Files
react-native/packager/src/node-haste/__tests__/DependencyGraph-test.js
Christoph Pojer 657ea49bd6 Properly mock path in DependencyGraph-test
Reviewed By: jeanlauliac

Differential Revision: D5111337

fbshipit-source-id: ceca2ec8917eda9ca34d65324d8b8c2a468eed1c
2017-05-23 05:18:23 -07:00

5822 lines
159 KiB
JavaScript

/**
* Copyright (c) 2015-present, Facebook, Inc.
* All rights reserved.
*
* This source code is licensed under the BSD-style license found in the
* LICENSE file in the root directory of this source tree. An additional grant
* of patent rights can be found in the PATENTS file in the same directory.
*
* @format
*/
'use strict';
jest.disableAutomock();
jest.useRealTimers();
jest
.mock('fs')
.mock('../../Logger')
.mock('../../lib/TransformCache')
// It's noticeably faster to prevent running watchman from FileWatcher.
.mock('child_process', () => ({}));
// This doesn't have state, and it's huge (Babel) so it's much faster to
// require it only once. The variable name is prefixed with "mock" as an escape-hatch
// for babel-plugin-jest-hoist.
const mockExtractDependencies = require('../../JSTransformer/worker/extract-dependencies');
jest.mock(
'../../JSTransformer/worker/extract-dependencies',
() => mockExtractDependencies,
);
jasmine.DEFAULT_TIMEOUT_INTERVAL = 10000;
beforeEach(() => {
jest.resetModules();
jest.mock('path', () => require.requireActual('path'));
});
describe('DependencyGraph', function() {
let Module;
let ResolutionRequest;
let defaults;
let emptyTransformOptions;
function getOrderedDependenciesAsJSON(
dgraphPromise,
entryPath,
platform,
recursive = true,
) {
return Promise.resolve(dgraphPromise)
.then(dgraph =>
dgraph.getDependencies({
entryPath,
options: emptyTransformOptions,
platform,
recursive,
}),
)
.then(response => response.finalize())
.then(({dependencies}) =>
Promise.all(
dependencies.map(dep =>
Promise.all([
dep.getName(),
dep.getDependencies(),
]).then(([name, moduleDependencies]) => ({
path: dep.path,
isJSON: dep.isJSON(),
isAsset: dep.isAsset(),
isPolyfill: dep.isPolyfill(),
resolution: dep.resolution,
id: name,
dependencies: moduleDependencies,
})),
),
),
);
}
beforeEach(function() {
jest.resetModules();
Module = require('../Module');
ResolutionRequest = require('../DependencyGraph/ResolutionRequest');
emptyTransformOptions = {transformer: {transform: {}}};
defaults = {
assetExts: ['png', 'jpg'],
forceNodeFilesystemAPI: true,
providesModuleNodeModules: ['haste-fbjs', 'react-haste', 'react-native'],
platforms: new Set(['ios', 'android']),
useWatchman: false,
ignoreFilePath: () => false,
maxWorkerCount: 1,
moduleOptions: {},
resetCache: true,
transformCode: (module, sourceCode, transformOptions) => {
return new Promise(resolve => {
let deps = {dependencies: [], dependencyOffsets: []};
if (!module.path.endsWith('.json')) {
deps = mockExtractDependencies(sourceCode);
}
resolve({...deps, code: sourceCode});
});
},
getTransformCacheKey: () => 'abcdef',
reporter: require('../../lib/reporting').nullReporter,
sourceExts: ['js', 'json'],
watch: true,
};
});
describe('get sync dependencies (posix)', function() {
let DependencyGraph;
const consoleWarn = console.warn;
const realPlatform = process.platform;
beforeEach(function() {
process.platform = 'linux';
DependencyGraph = require('../DependencyGraph');
});
afterEach(function() {
console.warn = consoleWarn;
process.platform = realPlatform;
});
it('should get dependencies', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("a")',
].join('\n'),
'a.js': ['/**', ' * @providesModule a', ' */', 'require("b")'].join(
'\n',
),
'b.js': ['/**', ' * @providesModule b', ' */'].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'a',
path: '/root/a.js',
dependencies: ['b'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'b',
path: '/root/b.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
});
it('should resolve relative entry path', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': ['/**', ' * @providesModule index', ' */'].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(dgraph, 'index.js').then(function(
deps,
) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should get shallow dependencies', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("a")',
].join('\n'),
'a.js': ['/**', ' * @providesModule a', ' */', 'require("b")'].join(
'\n',
),
'b.js': ['/**', ' * @providesModule b', ' */'].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
null,
false,
).then(deps => {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'a',
path: '/root/a.js',
dependencies: ['b'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should get dependencies with the correct extensions', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("a")',
].join('\n'),
'a.js': ['/**', ' * @providesModule a', ' */'].join('\n'),
'a.js.orig': ['/**', ' * @providesModule a', ' */'].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'a',
path: '/root/a.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should get json dependencies', function() {
var root = '/root';
setMockFileSystem({
root: {
'package.json': JSON.stringify({
name: 'package',
}),
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("./a.json")',
'require("./b")',
].join('\n'),
'a.json': JSON.stringify({}),
'b.json': JSON.stringify({}),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['./a.json', './b'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'package/a.json',
isJSON: true,
path: '/root/a.json',
dependencies: [],
isAsset: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'package/b.json',
isJSON: true,
path: '/root/b.json',
dependencies: [],
isAsset: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should get package json as a dep', () => {
var root = '/root';
setMockFileSystem({
root: {
'package.json': JSON.stringify({
name: 'package',
}),
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("./package.json")',
].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(deps => {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['./package.json'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'package/package.json',
isJSON: true,
path: '/root/package.json',
dependencies: [],
isAsset: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should get dependencies with relative assets', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("./imgs/a.png")',
].join('\n'),
imgs: {
'a.png': '',
},
'package.json': JSON.stringify({
name: 'rootPackage',
}),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['./imgs/a.png'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'rootPackage/imgs/a.png',
path: '/root/imgs/a.png',
dependencies: [],
isAsset: true,
resolution: 1,
isJSON: false,
isPolyfill: false,
},
]);
});
});
it('should get dependencies with assets and resolution', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("./imgs/a.png");',
'require("./imgs/b.png");',
'require("./imgs/c.png");',
].join('\n'),
imgs: {
'a@1.5x.png': '',
'b@.7x.png': '',
'c.png': '',
'c@2x.png': '',
},
'package.json': JSON.stringify({
name: 'rootPackage',
}),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['./imgs/a.png', './imgs/b.png', './imgs/c.png'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'rootPackage/imgs/a.png',
path: '/root/imgs/a@1.5x.png',
resolution: 1.5,
dependencies: [],
isAsset: true,
isJSON: false,
isPolyfill: false,
},
{
id: 'rootPackage/imgs/b.png',
path: '/root/imgs/b@.7x.png',
resolution: 0.7,
dependencies: [],
isAsset: true,
isJSON: false,
isPolyfill: false,
},
{
id: 'rootPackage/imgs/c.png',
path: '/root/imgs/c.png',
resolution: 1,
dependencies: [],
isAsset: true,
isJSON: false,
isPolyfill: false,
},
]);
});
});
it('should respect platform extension in assets', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("./imgs/a.png");',
'require("./imgs/b.png");',
'require("./imgs/c.png");',
].join('\n'),
imgs: {
'a@1.5x.ios.png': '',
'b@.7x.ios.png': '',
'c.ios.png': '',
'c@2x.ios.png': '',
},
'package.json': JSON.stringify({
name: 'rootPackage',
}),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
'ios',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['./imgs/a.png', './imgs/b.png', './imgs/c.png'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'rootPackage/imgs/a.png',
path: '/root/imgs/a@1.5x.ios.png',
resolution: 1.5,
dependencies: [],
isAsset: true,
isJSON: false,
isPolyfill: false,
},
{
id: 'rootPackage/imgs/b.png',
path: '/root/imgs/b@.7x.ios.png',
resolution: 0.7,
dependencies: [],
isAsset: true,
isJSON: false,
isPolyfill: false,
},
{
id: 'rootPackage/imgs/c.png',
path: '/root/imgs/c.ios.png',
resolution: 1,
dependencies: [],
isAsset: true,
isJSON: false,
isPolyfill: false,
},
]);
});
});
it('should get recursive dependencies', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("a")',
].join('\n'),
'a.js': [
'/**',
' * @providesModule a',
' */',
'require("index")',
].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'a',
path: '/root/a.js',
dependencies: ['index'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with packages', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'main.js',
}),
'main.js': 'lol',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/main.js',
path: '/root/aPackage/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with packages with a trailing slash', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage/")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'main.js',
}),
'main.js': 'lol',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage/'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/main.js',
path: '/root/aPackage/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with packages with a dot in the name', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("sha.js")',
'require("x.y.z")',
].join('\n'),
'sha.js': {
'package.json': JSON.stringify({
name: 'sha.js',
main: 'main.js',
}),
'main.js': 'lol',
},
'x.y.z': {
'package.json': JSON.stringify({
name: 'x.y.z',
main: 'main.js',
}),
'main.js': 'lol',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['sha.js', 'x.y.z'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'sha.js/main.js',
path: '/root/sha.js/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'x.y.z/main.js',
path: '/root/x.y.z/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should default main package to index.js', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': 'require("aPackage")',
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
}),
'index.js': 'lol',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: '/root/index.js',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/index.js',
path: '/root/aPackage/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should resolve using alternative ids', () => {
var root = '/root';
setMockFileSystem({
root: {
'index.js': 'require("aPackage")',
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
}),
'index.js': ['/**', ' * @providesModule EpicModule', ' */'].join(
'\n',
),
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: '/root/index.js',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'EpicModule',
path: '/root/aPackage/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should default use index.js if main is a dir', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': 'require("aPackage")',
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'lib',
}),
lib: {
'index.js': 'lol',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: '/root/index.js',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/lib/index.js',
path: '/root/aPackage/lib/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should resolve require to index if it is a dir', function() {
var root = '/root';
setMockFileSystem({
root: {
'package.json': JSON.stringify({
name: 'test',
}),
'index.js': 'require("./lib/")',
lib: {
'index.js': 'lol',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'test/index.js',
path: '/root/index.js',
dependencies: ['./lib/'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'test/lib/index.js',
path: '/root/lib/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should resolve require to main if it is a dir w/ a package.json', function() {
var root = '/root';
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 = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'test/index.js',
path: '/root/index.js',
dependencies: ['./lib/'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: '/root/lib/main.js',
path: '/root/lib/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should ignore malformed packages', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': ['/**', ' * @providesModule index', ' */'].join('\n'),
aPackage: {
'package.json': 'lol',
'main.js': 'lol',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should fatal on multiple modules with the same name', function() {
const root = '/root';
console.warn = jest.fn();
setMockFileSystem({
root: {
'index.js': ['/**', ' * @providesModule index', ' */'].join('\n'),
'b.js': ['/**', ' * @providesModule index', ' */'].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return dgraph.catch(err => {
expect(err.message).toEqual(
`Failed to build DependencyGraph: @providesModule naming collision:\n` +
` Duplicate module name: index\n` +
` Paths: /root/b.js collides with /root/index.js\n\n` +
'This error is caused by a @providesModule declaration ' +
'with the same name across two different files.',
);
expect(err.type).toEqual('DependencyGraphError');
expect(console.warn).toBeCalled();
});
});
it('throws when a module is missing', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("lolomg")',
].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).catch(error => {
expect(error.type).toEqual('UnableToResolveError');
});
});
it('should work with packages with subdirs', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage/subdir/lolynot")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'main.js',
}),
'main.js': 'lol',
subdir: {
'lolynot.js': 'lolynot',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage/subdir/lolynot'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'aPackage/subdir/lolynot.js',
path: '/root/aPackage/subdir/lolynot.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
});
it('should work with relative modules in packages', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'main.js',
}),
'main.js': 'require("./subdir/lolynot")',
subdir: {
'lolynot.js': 'require("../other")',
},
'other.js': '/* some code */',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'aPackage/main.js',
path: '/root/aPackage/main.js',
dependencies: ['./subdir/lolynot'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'aPackage/subdir/lolynot.js',
path: '/root/aPackage/subdir/lolynot.js',
dependencies: ['../other'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'aPackage/other.js',
path: '/root/aPackage/other.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
});
testBrowserField('browser');
testBrowserField('react-native');
function replaceBrowserField(json, fieldName) {
if (fieldName !== 'browser') {
json[fieldName] = json.browser;
delete json.browser;
}
return json;
}
function testBrowserField(fieldName) {
it(
'should support simple browser field in packages ("' + fieldName + '")',
function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify(
replaceBrowserField(
{
name: 'aPackage',
main: 'main.js',
browser: 'client.js',
},
fieldName,
),
),
'main.js': 'some other code',
'client.js': '/* some code */',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'aPackage/client.js',
path: '/root/aPackage/client.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
},
);
it(
'should support browser field in packages w/o .js ext ("' +
fieldName +
'")',
function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify(
replaceBrowserField(
{
name: 'aPackage',
main: 'main.js',
browser: 'client',
},
fieldName,
),
),
'main.js': 'some other code',
'client.js': '/* some code */',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/client.js',
path: '/root/aPackage/client.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
},
);
it(
'should support mapping main in browser field json ("' +
fieldName +
'")',
function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify(
replaceBrowserField(
{
name: 'aPackage',
main: './main.js',
browser: {
'./main.js': './client.js',
},
},
fieldName,
),
),
'main.js': 'some other code',
'client.js': '/* some code */',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
assetExts: ['png', 'jpg'],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/client.js',
path: '/root/aPackage/client.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
},
);
it(
'should work do correct browser mapping w/o js ext ("' +
fieldName +
'")',
function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify(
replaceBrowserField(
{
name: 'aPackage',
main: './main.js',
browser: {
'./main': './client.js',
},
},
fieldName,
),
),
'main.js': 'some other code',
'client.js': '/* some code */',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
assetExts: ['png', 'jpg'],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'aPackage/client.js',
path: '/root/aPackage/client.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
},
);
it(
'should support browser mapping of files ("' + fieldName + '")',
function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify(
replaceBrowserField(
{
name: 'aPackage',
main: './main.js',
browser: {
'./main': './client.js',
'./node.js': './not-node.js',
'./not-browser': './browser.js',
'./dir/server.js': './dir/client',
'./hello.js': './bye.js',
},
},
fieldName,
),
),
'main.js': '/* some other code */',
'client.js': 'require("./node")\nrequire("./dir/server.js")',
'not-node.js': 'require("./not-browser")',
'not-browser.js': 'require("./dir/server")',
'browser.js': '/* some browser code */',
dir: {
'server.js': '/* some node code */',
'client.js': 'require("../hello")',
},
'hello.js': '/* hello */',
'bye.js': '/* bye */',
},
},
});
const dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/client.js',
path: '/root/aPackage/client.js',
dependencies: ['./node', './dir/server.js'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/not-node.js',
path: '/root/aPackage/not-node.js',
dependencies: ['./not-browser'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/browser.js',
path: '/root/aPackage/browser.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/dir/client.js',
path: '/root/aPackage/dir/client.js',
dependencies: ['../hello'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/bye.js',
path: '/root/aPackage/bye.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
},
);
it(
'should support browser mapping for packages ("' + fieldName + '")',
function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify(
replaceBrowserField(
{
name: 'aPackage',
browser: {
'node-package': 'browser-package',
},
},
fieldName,
),
),
'index.js': 'require("node-package")',
'node-package': {
'package.json': JSON.stringify({
name: 'node-package',
}),
'index.js': '/* some node code */',
},
'browser-package': {
'package.json': JSON.stringify({
name: 'browser-package',
}),
'index.js': '/* some browser code */',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/index.js',
path: '/root/aPackage/index.js',
dependencies: ['node-package'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'browser-package/index.js',
path: '/root/aPackage/browser-package/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
},
);
it(
'should support browser mapping of a package to a file ("' +
fieldName +
'")',
() => {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify(
replaceBrowserField(
{
name: 'aPackage',
browser: {
'node-package': './dir/browser.js',
},
},
fieldName,
),
),
'index.js': 'require("./dir/ooga")',
dir: {
'ooga.js': 'require("node-package")',
'browser.js': '/* some browser code */',
},
'node-package': {
'package.json': JSON.stringify({
name: 'node-package',
}),
'index.js': '/* some node code */',
},
},
},
});
const dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/index.js',
path: '/root/aPackage/index.js',
dependencies: ['./dir/ooga'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/dir/ooga.js',
path: '/root/aPackage/dir/ooga.js',
dependencies: ['node-package'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/dir/browser.js',
path: '/root/aPackage/dir/browser.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
},
);
it(
'should support browser mapping for packages ("' + fieldName + '")',
function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify(
replaceBrowserField(
{
name: 'aPackage',
browser: {
'node-package': 'browser-package',
},
},
fieldName,
),
),
'index.js': 'require("node-package")',
'node-package': {
'package.json': JSON.stringify({
name: 'node-package',
}),
'index.js': '/* some node code */',
},
'browser-package': {
'package.json': JSON.stringify({
name: 'browser-package',
}),
'index.js': '/* some browser code */',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/index.js',
path: '/root/aPackage/index.js',
dependencies: ['node-package'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'browser-package/index.js',
path: '/root/aPackage/browser-package/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
},
);
it(
'should support browser exclude of a package ("' + fieldName + '")',
function() {
ResolutionRequest.EMPTY_MODULE = '/root/emptyModule.js';
var root = '/root';
setMockFileSystem({
root: {
'emptyModule.js': '',
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify(
replaceBrowserField(
{
name: 'aPackage',
browser: {
booga: false,
},
},
fieldName,
),
),
'index.js': 'require("booga")',
booga: {
'package.json': JSON.stringify({
name: 'booga',
}),
'index.js': '/* some node code */',
},
},
},
});
const dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/index.js',
path: '/root/aPackage/index.js',
dependencies: ['booga'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
dependencies: [],
id: '/root/emptyModule.js',
isAsset: false,
isJSON: false,
isPolyfill: false,
path: '/root/emptyModule.js',
resolution: undefined,
},
]);
});
},
);
it(
'should support browser exclude of a file ("' + fieldName + '")',
function() {
ResolutionRequest.EMPTY_MODULE = '/root/emptyModule.js';
var root = '/root';
setMockFileSystem({
root: {
'emptyModule.js': '',
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify(
replaceBrowserField(
{
name: 'aPackage',
browser: {
'./booga.js': false,
},
},
fieldName,
),
),
'index.js': 'require("./booga")',
'booga.js': '/* some node code */',
},
},
});
const dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/index.js',
path: '/root/aPackage/index.js',
dependencies: ['./booga'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
dependencies: [],
id: '/root/emptyModule.js',
isAsset: false,
isJSON: false,
isPolyfill: false,
path: '/root/emptyModule.js',
resolution: undefined,
},
]);
});
},
);
}
it('should fall back to browser mapping from react-native mapping', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
'react-native': {
'node-package': 'rn-package',
},
}),
'index.js': 'require("node-package")',
node_modules: {
'node-package': {
'package.json': JSON.stringify({
name: 'node-package',
}),
'index.js': '/* some node code */',
},
'rn-package': {
'package.json': JSON.stringify({
name: 'rn-package',
browser: {
'nested-package': 'nested-browser-package',
},
}),
'index.js': 'require("nested-package")',
},
'nested-browser-package': {
'package.json': JSON.stringify({
name: 'nested-browser-package',
}),
'index.js': '/* some code */',
},
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/index.js',
path: '/root/aPackage/index.js',
dependencies: ['node-package'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'rn-package/index.js',
path: '/root/aPackage/node_modules/rn-package/index.js',
dependencies: ['nested-package'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'nested-browser-package/index.js',
path: '/root/aPackage/node_modules/nested-browser-package/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with absolute paths', () => {
const root = '/root';
setMockFileSystem({
[root.slice(1)]: {
'index.js': 'require("/root/apple.js");',
'apple.js': '',
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: '/root/index.js',
path: '/root/index.js',
dependencies: ['/root/apple.js'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: '/root/apple.js',
path: '/root/apple.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should merge browser mapping with react-native mapping', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
'react-native': {
// should see this:
'node-package-a': 'rn-package-a',
// should see this:
'node-package-c': 'rn-package-d',
},
browser: {
// should see this:
'node-package-b': 'rn-package-b',
// should NOT see this:
'node-package-c': 'rn-package-c',
},
}),
'index.js': 'require("node-package-a"); require("node-package-b"); require("node-package-c");',
node_modules: {
'node-package-a': {
'package.json': JSON.stringify({
name: 'node-package-a',
}),
'index.js': '/* some node code */',
},
'node-package-b': {
'package.json': JSON.stringify({
name: 'node-package-b',
}),
'index.js': '/* some node code */',
},
'node-package-c': {
'package.json': JSON.stringify({
name: 'node-package-c',
}),
'index.js': '/* some node code */',
},
'node-package-d': {
'package.json': JSON.stringify({
name: 'node-package-d',
}),
'index.js': '/* some node code */',
},
'rn-package-a': {
'package.json': JSON.stringify({
name: 'rn-package-a',
}),
'index.js': '/* some rn code */',
},
'rn-package-b': {
'package.json': JSON.stringify({
name: 'rn-package-b',
}),
'index.js': '/* some rn code */',
},
'rn-package-c': {
'package.json': JSON.stringify({
name: 'rn-package-c',
}),
'index.js': '/* some rn code */',
},
'rn-package-d': {
'package.json': JSON.stringify({
name: 'rn-package-d',
}),
'index.js': '/* some rn code */',
},
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/index.js',
path: '/root/aPackage/index.js',
dependencies: [
'node-package-a',
'node-package-b',
'node-package-c',
],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'rn-package-a/index.js',
path: '/root/aPackage/node_modules/rn-package-a/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'rn-package-b/index.js',
path: '/root/aPackage/node_modules/rn-package-b/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'rn-package-d/index.js',
path: '/root/aPackage/node_modules/rn-package-d/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should fall back to `extraNodeModules`', () => {
const root = '/root';
setMockFileSystem({
[root.slice(1)]: {
'index.js': 'require("./foo")',
foo: {
'index.js': 'require("bar")',
},
'provides-bar': {
'package.json': '{"main": "lib/bar.js"}',
lib: {
'bar.js': '',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
extraNodeModules: {
bar: root + '/provides-bar',
},
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(deps => {
expect(deps).toEqual([
{
id: '/root/index.js',
path: '/root/index.js',
dependencies: ['./foo'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: '/root/foo/index.js',
path: '/root/foo/index.js',
dependencies: ['bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: '/root/provides-bar/lib/bar.js',
path: '/root/provides-bar/lib/bar.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should only use `extraNodeModules` after checking all possible filesystem locations', () => {
const root = '/root';
setMockFileSystem({
[root.slice(1)]: {
'index.js': 'require("bar")',
node_modules: {'bar.js': ''},
'provides-bar': {'index.js': ''},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
extraNodeModules: {
bar: root + '/provides-bar',
},
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(deps => {
expect(deps).toEqual([
{
id: '/root/index.js',
path: '/root/index.js',
dependencies: ['bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: '/root/node_modules/bar.js',
path: '/root/node_modules/bar.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should be able to resolve paths within `extraNodeModules`', () => {
const root = '/root';
setMockFileSystem({
[root.slice(1)]: {
'index.js': 'require("bar/lib/foo")',
'provides-bar': {
'package.json': '{}',
lib: {'foo.js': ''},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
extraNodeModules: {
bar: root + '/provides-bar',
},
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(deps => {
expect(deps).toEqual([
{
id: '/root/index.js',
path: '/root/index.js',
dependencies: ['bar/lib/foo'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: '/root/provides-bar/lib/foo.js',
path: '/root/provides-bar/lib/foo.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
});
describe('get sync dependencies (win32)', function() {
const realPlatform = process.platform;
let DependencyGraph;
beforeEach(function() {
process.platform = 'win32';
// reload path module
jest.resetModules();
jest.mock('path', () => require.requireActual('path').win32);
DependencyGraph = require('../DependencyGraph');
});
afterEach(function() {
process.platform = realPlatform;
});
it('should get dependencies', function() {
const root = 'C:\\root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("a")',
].join('\n'),
'a.js': ['/**', ' * @providesModule a', ' */', 'require("b")'].join(
'\n',
),
'b.js': ['/**', ' * @providesModule b', ' */'].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'C:\\root\\index.js',
).then(deps => {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.js',
dependencies: ['a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'a',
path: 'C:\\root\\a.js',
dependencies: ['b'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'b',
path: 'C:\\root\\b.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
});
it('should work with absolute paths', () => {
const root = 'C:\\root';
setMockFileSystem({
root: {
'index.js': 'require("C:/root/apple.js");',
'apple.js': '',
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'C:\\root\\index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'C:\\root\\index.js',
path: 'C:\\root\\index.js',
dependencies: ['C:/root/apple.js'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'C:\\root\\apple.js',
path: 'C:\\root\\apple.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should get dependencies with assets and resolution', function() {
const root = 'C:\\root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("./imgs/a.png");',
'require("./imgs/b.png");',
'require("./imgs/c.png");',
].join('\n'),
imgs: {
'a@1.5x.png': '',
'b@.7x.png': '',
'c.png': '',
'c@2x.png': '',
},
'package.json': JSON.stringify({
name: 'rootPackage',
}),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'C:\\root\\index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.js',
dependencies: ['./imgs/a.png', './imgs/b.png', './imgs/c.png'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'rootPackage/imgs/a.png',
path: 'C:\\root\\imgs\\a@1.5x.png',
resolution: 1.5,
dependencies: [],
isAsset: true,
isJSON: false,
isPolyfill: false,
},
{
id: 'rootPackage/imgs/b.png',
path: 'C:\\root\\imgs\\b@.7x.png',
resolution: 0.7,
dependencies: [],
isAsset: true,
isJSON: false,
isPolyfill: false,
},
{
id: 'rootPackage/imgs/c.png',
path: 'C:\\root\\imgs\\c.png',
resolution: 1,
dependencies: [],
isAsset: true,
isJSON: false,
isPolyfill: false,
},
]);
});
});
});
describe('node_modules (posix)', function() {
const realPlatform = process.platform;
let DependencyGraph;
beforeEach(function() {
process.platform = 'linux';
DependencyGraph = require('../DependencyGraph');
});
afterEach(function() {
process.platform = realPlatform;
});
it('should work with nested node_modules', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo");',
'require("bar");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': 'require("bar");\n/* foo module */',
node_modules: {
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': '/* bar 1 module */',
},
},
},
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': '/* bar 2 module */',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['foo', 'bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/main.js',
path: '/root/node_modules/foo/main.js',
dependencies: ['bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main.js',
path: '/root/node_modules/foo/node_modules/bar/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main.js',
path: '/root/node_modules/bar/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('platform should work with node_modules', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.ios.js': [
'/**',
' * @providesModule index',
' */',
'require("foo");',
'require("bar");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
}),
'index.ios.js': '',
},
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main',
}),
'main.ios.js': '',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.ios.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.ios.js',
dependencies: ['foo', 'bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/index.ios.js',
path: '/root/node_modules/foo/index.ios.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main.ios.js',
path: '/root/node_modules/bar/main.ios.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('nested node_modules with specific paths', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo");',
'require("bar/");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': 'require("bar/lol");\n/* foo module */',
node_modules: {
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': '/* bar 1 module */',
'lol.js': '',
},
},
},
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': '/* bar 2 module */',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['foo', 'bar/'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/main.js',
path: '/root/node_modules/foo/main.js',
dependencies: ['bar/lol'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/lol.js',
path: '/root/node_modules/foo/node_modules/bar/lol.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main.js',
path: '/root/node_modules/bar/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('nested node_modules with browser field', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo");',
'require("bar");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': 'require("bar/lol");\n/* foo module */',
node_modules: {
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
browser: {
'./lol': './wow',
},
}),
'main.js': '/* bar 1 module */',
'lol.js': '',
'wow.js': '',
},
},
},
bar: {
'package.json': JSON.stringify({
name: 'bar',
browser: './main2',
}),
'main2.js': '/* bar 2 module */',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['foo', 'bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/main.js',
path: '/root/node_modules/foo/main.js',
dependencies: ['bar/lol'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/lol.js',
path: '/root/node_modules/foo/node_modules/bar/lol.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main2.js',
path: '/root/node_modules/bar/main2.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('node_modules should support multi level', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("bar");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': '',
},
},
path: {
to: {
'bar.js': [
'/**',
' * @providesModule bar',
' */',
'require("foo")',
].join('\n'),
},
node_modules: {},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar',
path: '/root/path/to/bar.js',
dependencies: ['foo'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/main.js',
path: '/root/node_modules/foo/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should selectively ignore providesModule in node_modules', function() {
var root = '/root';
var otherRoot = '/anotherRoot';
const filesystem = {
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("shouldWork");',
'require("dontWork");',
'require("wontWork");',
'require("ember");',
'require("internalVendoredPackage");',
'require("anotherIndex");',
].join('\n'),
node_modules: {
'react-haste': {
'package.json': JSON.stringify({
name: 'react-haste',
main: 'main.js',
}),
// @providesModule should not be ignored here, because react-haste is whitelisted
'main.js': [
'/**',
' * @providesModule shouldWork',
' */',
'require("submodule");',
].join('\n'),
node_modules: {
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
// @providesModule should be ignored here, because it's not whitelisted
'main.js': [
'/**',
' * @providesModule dontWork',
' */',
'hi();',
].join('\n'),
},
submodule: {
'package.json': JSON.stringify({
name: 'submodule',
main: 'main.js',
}),
'main.js': 'log()',
},
},
},
ember: {
'package.json': JSON.stringify({
name: 'ember',
main: 'main.js',
}),
// @providesModule should be ignored here, because it's not whitelisted,
// and also, the modules "id" should be ember/main.js, not it's haste name
'main.js': [
'/**',
' * @providesModule wontWork',
' */',
'hi();',
].join('\n'),
},
},
// This part of the dep graph is meant to emulate internal facebook infra.
// By whitelisting `vendored_modules`, haste should still work.
vendored_modules: {
'a-vendored-package': {
'package.json': JSON.stringify({
name: 'a-vendored-package',
main: 'main.js',
}),
// @providesModule should _not_ be ignored here, because it's whitelisted.
'main.js': [
'/**',
' * @providesModule internalVendoredPackage',
' */',
'hiFromInternalPackage();',
].join('\n'),
},
},
},
// we need to support multiple roots and using haste between them
anotherRoot: {
'index.js': [
'/**',
' * @providesModule anotherIndex',
' */',
'wazup()',
].join('\n'),
},
};
setMockFileSystem(filesystem);
var dgraph = DependencyGraph.load({
...defaults,
roots: [root, otherRoot],
});
return getOrderedDependenciesAsJSON(dgraph, '/root/index.js')
.catch(error => {
expect(error.type).toEqual('UnableToResolveError');
})
.then(() => {
filesystem.root['index.js'] = filesystem.root['index.js']
.replace('require("dontWork")', '')
.replace('require("wontWork")', '');
return triggerAndProcessWatchEvent(
dgraph,
'change',
root + '/index.js',
)
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.then(deps => {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: [
'shouldWork',
'ember',
'internalVendoredPackage',
'anotherIndex',
],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'shouldWork',
path: '/root/node_modules/react-haste/main.js',
dependencies: ['submodule'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'submodule/main.js',
path: '/root/node_modules/react-haste/node_modules/submodule/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'ember/main.js',
path: '/root/node_modules/ember/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'internalVendoredPackage',
path: '/root/vendored_modules/a-vendored-package/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'anotherIndex',
path: '/anotherRoot/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
});
it('should not be confused by prev occuring whitelisted names', function() {
var root = '/react-haste';
setMockFileSystem({
'react-haste': {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("shouldWork");',
].join('\n'),
node_modules: {
'react-haste': {
'package.json': JSON.stringify({
name: 'react-haste',
main: 'main.js',
}),
'main.js': ['/**', ' * @providesModule shouldWork', ' */'].join(
'\n',
),
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/react-haste/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/react-haste/index.js',
dependencies: ['shouldWork'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'shouldWork',
path: '/react-haste/node_modules/react-haste/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with node packages with a .js in the name', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("sha.js")',
].join('\n'),
node_modules: {
'sha.js': {
'package.json': JSON.stringify({
name: 'sha.js',
main: 'main.js',
}),
'main.js': 'lol',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['sha.js'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'sha.js/main.js',
path: '/root/node_modules/sha.js/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with multiple platforms (haste)', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.ios.js': `
/**
* @providesModule index
*/
require('a');
`,
'a.ios.js': `
/**
* @providesModule a
*/
`,
'a.android.js': `
/**
* @providesModule a
*/
`,
'a.js': `
/**
* @providesModule a
*/
`,
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.ios.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.ios.js',
dependencies: ['a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'a',
path: '/root/a.ios.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should pick the generic file', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.ios.js': `
/**
* @providesModule index
*/
require('a');
`,
'a.android.js': `
/**
* @providesModule a
*/
`,
'a.js': `
/**
* @providesModule a
*/
`,
'a.web.js': `
/**
* @providesModule a
*/
`,
},
});
var dgraph = DependencyGraph.load({
...defaults,
platforms: new Set(['ios', 'android', 'web']),
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.ios.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.ios.js',
dependencies: ['a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'a',
path: '/root/a.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with multiple platforms (node)', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.ios.js': `
/**
* @providesModule index
*/
require('./a');
`,
'a.ios.js': '',
'a.android.js': '',
'a.js': '',
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.ios.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.ios.js',
dependencies: ['./a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: '/root/a.ios.js',
path: '/root/a.ios.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should require package.json', () => {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo/package.json");',
'require("bar");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
},
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': 'require("./package.json")',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(deps => {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['foo/package.json', 'bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/package.json',
path: '/root/node_modules/foo/package.json',
dependencies: [],
isAsset: false,
isJSON: true,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main.js',
path: '/root/node_modules/bar/main.js',
dependencies: ['./package.json'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/package.json',
path: '/root/node_modules/bar/package.json',
dependencies: [],
isAsset: false,
isJSON: true,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with one-character node_modules', () => {
const root = '/root';
setMockFileSystem({
[root.slice(1)]: {
'index.js': 'require("a/index.js");',
node_modules: {
a: {
'package.json': '{"name": "a", "version": "1.2.3"}',
'index.js': '',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: '/root/index.js',
path: '/root/index.js',
dependencies: ['a/index.js'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'a/index.js',
path: '/root/node_modules/a/index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
});
describe('node_modules (win32)', function() {
const realPlatform = process.platform;
// these tests will not work in a simulated way on linux testing VMs
// due to the drive letter expectation
if (realPlatform !== 'win32') {
return;
}
const DependencyGraph = require('../DependencyGraph');
it('should work with nested node_modules', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo");',
'require("bar");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': 'require("bar");\n/* foo module */',
node_modules: {
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': '/* bar 1 module */',
},
},
},
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': '/* bar 2 module */',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.js',
dependencies: ['foo', 'bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/main.js',
path: 'C:\\root\\node_modules\\foo\\main.js',
dependencies: ['bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main.js',
path: 'C:\\root\\node_modules\\foo\\node_modules\\bar\\main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main.js',
path: 'C:\\root\\node_modules\\bar\\main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('platform should work with node_modules', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.ios.js': [
'/**',
' * @providesModule index',
' */',
'require("foo");',
'require("bar");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
}),
'index.ios.js': '',
},
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main',
}),
'main.ios.js': '',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.ios.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.ios.js',
dependencies: ['foo', 'bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/index.ios.js',
path: 'C:\\root\\node_modules\\foo\\index.ios.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main.ios.js',
path: 'C:\\root\\node_modules\\bar\\main.ios.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('nested node_modules with specific paths', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo");',
'require("bar/");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': 'require("bar/lol");\n/* foo module */',
node_modules: {
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': '/* bar 1 module */',
'lol.js': '',
},
},
},
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': '/* bar 2 module */',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.js',
dependencies: ['foo', 'bar/'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/main.js',
path: 'C:\\root\\node_modules\\foo\\main.js',
dependencies: ['bar/lol'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/lol.js',
path: 'C:\\root\\node_modules\\foo\\node_modules\\bar\\lol.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main.js',
path: 'C:\\root\\node_modules\\bar\\main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('nested node_modules with browser field', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo");',
'require("bar");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': 'require("bar/lol");\n/* foo module */',
node_modules: {
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
browser: {
'./lol': './wow',
},
}),
'main.js': '/* bar 1 module */',
'lol.js': '',
'wow.js': '',
},
},
},
bar: {
'package.json': JSON.stringify({
name: 'bar',
browser: './main2',
}),
'main2.js': '/* bar 2 module */',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.js',
dependencies: ['foo', 'bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/main.js',
path: 'C:\\root\\node_modules\\foo\\main.js',
dependencies: ['bar/lol'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/lol.js',
path: 'C:\\root\\node_modules\\foo\\node_modules\\bar\\lol.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main2.js',
path: 'C:\\root\\node_modules\\bar\\main2.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('node_modules should support multi level', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("bar");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': '',
},
},
path: {
to: {
'bar.js': [
'/**',
' * @providesModule bar',
' */',
'require("foo")',
].join('\n'),
},
node_modules: {},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.js',
dependencies: ['bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar',
path: 'C:\\root\\path\\to\\bar.js',
dependencies: ['foo'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/main.js',
path: 'C:\\root\\node_modules\\foo\\main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should selectively ignore providesModule in node_modules', function() {
var root = '/root';
var otherRoot = '/anotherRoot';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("shouldWork");',
'require("dontWork");',
'require("wontWork");',
'require("ember");',
'require("internalVendoredPackage");',
'require("anotherIndex");',
].join('\n'),
node_modules: {
'react-haste': {
'package.json': JSON.stringify({
name: 'react-haste',
main: 'main.js',
}),
// @providesModule should not be ignored here, because react-haste is whitelisted
'main.js': [
'/**',
' * @providesModule shouldWork',
' */',
'require("submodule");',
].join('\n'),
node_modules: {
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
// @providesModule should be ignored here, because it's not whitelisted
'main.js': [
'/**',
' * @providesModule dontWork',
' */',
'hi();',
].join('\n'),
},
submodule: {
'package.json': JSON.stringify({
name: 'submodule',
main: 'main.js',
}),
'main.js': 'log()',
},
},
},
ember: {
'package.json': JSON.stringify({
name: 'ember',
main: 'main.js',
}),
// @providesModule should be ignored here, because it's not whitelisted,
// and also, the modules "id" should be ember/main.js, not it's haste name
'main.js': [
'/**',
' * @providesModule wontWork',
' */',
'hi();',
].join('\n'),
},
},
// This part of the dep graph is meant to emulate internal facebook infra.
// By whitelisting `vendored_modules`, haste should still work.
vendored_modules: {
'a-vendored-package': {
'package.json': JSON.stringify({
name: 'a-vendored-package',
main: 'main.js',
}),
// @providesModule should _not_ be ignored here, because it's whitelisted.
'main.js': [
'/**',
' * @providesModule internalVendoredPackage',
' */',
'hiFromInternalPackage();',
].join('\n'),
},
},
},
// we need to support multiple roots and using haste between them
anotherRoot: {
'index.js': [
'/**',
' * @providesModule anotherIndex',
' */',
'wazup()',
].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root, otherRoot],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.js',
dependencies: [
'shouldWork',
'dontWork',
'wontWork',
'ember',
'internalVendoredPackage',
'anotherIndex',
],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'shouldWork',
path: 'C:\\root\\node_modules\\react-haste\\main.js',
dependencies: ['submodule'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'submodule/main.js',
path: 'C:\\root\\node_modules\\react-haste\\node_modules\\submodule\\main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'ember/main.js',
path: 'C:\\root\\node_modules\\ember\\main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'internalVendoredPackage',
path: 'C:\\root\\vendored_modules\\a-vendored-package\\main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'anotherIndex',
path: 'C:\\anotherRoot\\index.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should not be confused by prev occuring whitelisted names', function() {
var root = '/react-haste';
setMockFileSystem({
'react-haste': {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("shouldWork");',
].join('\n'),
node_modules: {
'react-haste': {
'package.json': JSON.stringify({
name: 'react-haste',
main: 'main.js',
}),
'main.js': ['/**', ' * @providesModule shouldWork', ' */'].join(
'\n',
),
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/react-haste/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\react-haste\\index.js',
dependencies: ['shouldWork'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'shouldWork',
path: 'C:\\react-haste\\node_modules\\react-haste\\main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should ignore modules it cant find (assumes own require system)', function() {
// For example SourceMap.js implements it's own require system.
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo/lol");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': '/* foo module */',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.js',
dependencies: ['foo/lol'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with node packages with a .js in the name', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("sha.js")',
].join('\n'),
node_modules: {
'sha.js': {
'package.json': JSON.stringify({
name: 'sha.js',
main: 'main.js',
}),
'main.js': 'lol',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.js',
dependencies: ['sha.js'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'sha.js/main.js',
path: 'C:\\root\\node_modules\\sha.js\\main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with multiple platforms (haste)', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.ios.js': `
/**
* @providesModule index
*/
require('a');
`,
'a.ios.js': `
/**
* @providesModule a
*/
`,
'a.android.js': `
/**
* @providesModule a
*/
`,
'a.js': `
/**
* @providesModule a
*/
`,
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.ios.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.ios.js',
dependencies: ['a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'a',
path: 'C:\\root\\a.ios.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should pick the generic file', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.ios.js': `
/**
* @providesModule index
*/
require('a');
`,
'a.android.js': `
/**
* @providesModule a
*/
`,
'a.js': `
/**
* @providesModule a
*/
`,
'a.web.js': `
/**
* @providesModule a
*/
`,
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.ios.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.ios.js',
dependencies: ['a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'a',
path: 'C:\\root\\a.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should work with multiple platforms (node)', function() {
var root = '/root';
setMockFileSystem({
root: {
'index.ios.js': `
/**
* @providesModule index
*/
require('./a');
`,
'a.ios.js': '',
'a.android.js': '',
'a.js': '',
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.ios.js',
).then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.ios.js',
dependencies: ['./a'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'C:\\root\\a.ios.js',
path: 'C:\\root\\a.ios.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
it('should require package.json', () => {
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo/package.json");',
'require("bar");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
},
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': 'require("./package.json")',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(deps => {
expect(deps).toEqual([
{
id: 'index',
path: 'C:\\root\\index.js',
dependencies: ['foo/package.json', 'bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'foo/package.json',
path: 'C:\\root\\node_modules\\foo\\package.json',
dependencies: [],
isAsset: false,
isJSON: true,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/main.js',
path: 'C:\\root\\node_modules\\bar\\main.js',
dependencies: ['./package.json'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar/package.json',
path: 'C:\\root\\node_modules\\bar\\package.json',
dependencies: [],
isAsset: false,
isJSON: true,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
});
describe('file watch updating', function() {
const realPlatform = process.platform;
let DependencyGraph;
beforeEach(function() {
process.platform = 'linux';
DependencyGraph = require('../DependencyGraph');
});
afterEach(function() {
process.platform = realPlatform;
});
it('updates module dependencies', function() {
var root = '/root';
var filesystem = setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
'require("foo")',
].join('\n'),
'foo.js': [
'/**',
' * @providesModule foo',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'main.js',
}),
'main.js': 'main',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function() {
filesystem.root['index.js'] = filesystem.root['index.js'].replace(
'require("foo")',
'',
);
return triggerAndProcessWatchEvent(dgraph, 'change', root + '/index.js')
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/main.js',
path: '/root/aPackage/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
});
it('updates module dependencies on file change', function() {
var root = '/root';
var filesystem = setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
'require("foo")',
].join('\n'),
'foo.js': [
'/**',
' * @providesModule foo',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'main.js',
}),
'main.js': 'main',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function() {
filesystem.root['index.js'] = filesystem.root['index.js'].replace(
'require("foo")',
'',
);
return triggerAndProcessWatchEvent(dgraph, 'change', root + '/index.js')
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/main.js',
path: '/root/aPackage/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
]);
});
});
});
it('updates module dependencies on file delete', function() {
expect.assertions(1);
var root = '/root';
var filesystem = setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
'require("foo")',
].join('\n'),
'foo.js': [
'/**',
' * @providesModule foo',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'main.js',
}),
'main.js': 'main',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(
dgraph,
'/root/index.js',
).then(function() {
delete filesystem.root['foo.js'];
return triggerAndProcessWatchEvent(dgraph, 'change', root + '/foo.js')
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.catch(error => expect(error.type).toEqual('UnableToResolveError'));
});
});
it('updates module dependencies on file add', function() {
expect.assertions(1);
var root = '/root';
var filesystem = setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
'require("foo")',
].join('\n'),
'foo.js': [
'/**',
' * @providesModule foo',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'main.js',
}),
'main.js': 'main',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(dgraph, '/root/index.js')
.then(function() {
filesystem.root['bar.js'] = [
'/**',
' * @providesModule bar',
' */',
'require("foo")',
].join('\n');
return triggerAndProcessWatchEvent(
dgraph,
'change',
root + '/bar.js',
);
})
.then(() => {
filesystem.root.aPackage['main.js'] = 'require("bar")';
return triggerAndProcessWatchEvent(
dgraph,
'change',
root + '/aPackage/main.js',
);
})
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage', 'foo'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'aPackage/main.js',
path: '/root/aPackage/main.js',
dependencies: ['bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
},
{
id: 'bar',
path: '/root/bar.js',
dependencies: ['foo'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'foo',
path: '/root/foo.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
});
it('updates module dependencies on relative asset add', function() {
var root = '/root';
var filesystem = setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("./foo.png")',
].join('\n'),
'package.json': JSON.stringify({
name: 'aPackage',
}),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
assetExts: ['png'],
});
return getOrderedDependenciesAsJSON(dgraph, '/root/index.js')
.catch(error => {
expect(error.type).toEqual('UnableToResolveError');
})
.then(() => {
filesystem.root['foo.png'] = '';
return triggerAndProcessWatchEvent(
dgraph,
'change',
root + '/foo.png',
);
})
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.then(function(deps2) {
expect(deps2).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['./foo.png'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'aPackage/foo.png',
path: '/root/foo.png',
dependencies: [],
isAsset: true,
resolution: 1,
isJSON: false,
isPolyfill: false,
resolveDependency: undefined,
},
]);
});
});
it('changes to browser field', function() {
expect.assertions(1);
var root = '/root';
var filesystem = setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'main.js',
}),
'main.js': 'main',
'browser.js': 'browser',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(dgraph, '/root/index.js')
.then(function() {
filesystem.root.aPackage['package.json'] = JSON.stringify({
name: 'aPackage',
main: 'main.js',
browser: 'browser.js',
});
return triggerAndProcessWatchEvent(
dgraph,
'change',
root + '/aPackage/package.json',
);
})
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['aPackage'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'aPackage/browser.js',
path: '/root/aPackage/browser.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
});
it('removes old package from cache', function() {
var root = '/root';
var filesystem = setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("aPackage")',
].join('\n'),
aPackage: {
'package.json': JSON.stringify({
name: 'aPackage',
main: 'main.js',
}),
'main.js': 'main',
'browser.js': 'browser',
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(dgraph, '/root/index.js')
.then(function() {
filesystem.root['index.js'] = [
'/**',
' * @providesModule index',
' */',
'require("bPackage")',
].join('\n');
filesystem.root.aPackage['package.json'] = JSON.stringify({
name: 'bPackage',
main: 'main.js',
});
return dgraph.then(
dg =>
new Promise(resolve => {
dg.once('change', () => resolve());
triggerWatchEvent('change', root + '/index.js');
triggerWatchEvent('change', root + '/aPackage/package.json');
}),
);
})
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.then(function(deps) {
expect(deps).toEqual([
{
dependencies: ['bPackage'],
id: 'index',
isAsset: false,
isJSON: false,
isPolyfill: false,
path: '/root/index.js',
resolution: undefined,
resolveDependency: undefined,
},
{
dependencies: [],
id: 'bPackage/main.js',
isAsset: false,
isJSON: false,
isPolyfill: false,
path: '/root/aPackage/main.js',
resolution: undefined,
},
]);
});
});
it('should update node package changes', function() {
expect.assertions(2);
var root = '/root';
var filesystem = setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': 'require("bar");\n/* foo module */',
node_modules: {
bar: {
'package.json': JSON.stringify({
name: 'bar',
main: 'main.js',
}),
'main.js': '/* bar 1 module */',
},
},
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(dgraph, '/root/index.js')
.then(function(deps) {
expect(deps).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['foo'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'foo/main.js',
path: '/root/node_modules/foo/main.js',
dependencies: ['bar'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'bar/main.js',
path: '/root/node_modules/foo/node_modules/bar/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
filesystem.root.node_modules.foo['main.js'] = 'lol';
return triggerAndProcessWatchEvent(
dgraph,
'change',
root + '/node_modules/foo/main.js',
);
})
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.then(function(deps2) {
expect(deps2).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['foo'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'foo/main.js',
path: '/root/node_modules/foo/main.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
});
it('should update node package main changes', function() {
expect.assertions(1);
var root = '/root';
var filesystem = setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'require("foo");',
].join('\n'),
node_modules: {
foo: {
'package.json': JSON.stringify({
name: 'foo',
main: 'main.js',
}),
'main.js': '/* foo module */',
'browser.js': '/* foo module */',
},
},
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(dgraph, '/root/index.js')
.then(function(deps) {
filesystem.root.node_modules.foo['package.json'] = JSON.stringify({
name: 'foo',
main: 'main.js',
browser: 'browser.js',
});
return triggerAndProcessWatchEvent(
dgraph,
'change',
root + '/node_modules/foo/package.json',
);
})
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.then(function(deps2) {
expect(deps2).toEqual([
{
id: 'index',
path: '/root/index.js',
dependencies: ['foo'],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: 'foo/browser.js',
path: '/root/node_modules/foo/browser.js',
dependencies: [],
isAsset: false,
isJSON: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
});
it('should not error when the watcher reports a known file as added', function() {
expect.assertions(1);
var root = '/root';
setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
'var b = require("b");',
].join('\n'),
'b.js': [
'/**',
' * @providesModule b',
' */',
'module.exports = function() {};',
].join('\n'),
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
});
return getOrderedDependenciesAsJSON(dgraph, '/root/index.js')
.then(() =>
triggerAndProcessWatchEvent(dgraph, 'change', root + '/index.js'),
)
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.js'))
.then(deps => {
expect(deps).toBeDefined();
});
});
it('should recover from multiple modules with the same name (but this is broken right now)', async () => {
const root = '/root';
console.warn = jest.fn();
const filesystem = setMockFileSystem({
root: {
'index.js': [
'/**',
' * @providesModule index',
' */',
"require('a')",
"require('b')",
].join('\n'),
'a.js': ['/**', ' * @providesModule a', ' */'].join('\n'),
'b.js': ['/**', ' * @providesModule b', ' */'].join('\n'),
},
});
const dgraph = DependencyGraph.load({...defaults, roots: [root]});
await getOrderedDependenciesAsJSON(dgraph, root + '/index.js');
filesystem.root['b.js'] = ['/**', ' * @providesModule a', ' */'].join(
'\n',
);
await triggerAndProcessWatchEvent(dgraph, 'change', root + '/b.js');
try {
await getOrderedDependenciesAsJSON(dgraph, root + '/index.js');
throw new Error('expected `getOrderedDependenciesAsJSON` to fail');
} catch (error) {
if (error.type !== 'UnableToResolveError') {
throw error;
}
expect(console.warn).toBeCalled();
filesystem.root['b.js'] = ['/**', ' * @providesModule b', ' */'].join(
'\n',
);
await triggerAndProcessWatchEvent(dgraph, 'change', root + '/b.js');
}
// This verifies that it is broken right now. Instead of throwing it should
// return correct results. Once this is fixed in `jest-haste`, remove
// the whole try catch and verify results are matching a snapshot.
try {
await getOrderedDependenciesAsJSON(dgraph, root + '/index.js');
throw new Error('expected `getOrderedDependenciesAsJSON` to fail');
} catch (error) {
if (error.type !== 'UnableToResolveError') {
throw error;
}
}
});
});
describe('Extensions', () => {
const realPlatform = process.platform;
let DependencyGraph;
beforeEach(function() {
process.platform = 'linux';
DependencyGraph = require('../DependencyGraph');
});
afterEach(function() {
process.platform = realPlatform;
});
it('supports custom file extensions', () => {
var root = '/root';
setMockFileSystem({
root: {
'index.jsx': [
'/**',
' * @providesModule index',
' */',
'require("a")',
].join('\n'),
'a.coffee': ['/**', ' * @providesModule a', ' */'].join('\n'),
'X.js': '',
},
});
var dgraph = DependencyGraph.load({
...defaults,
roots: [root],
sourceExts: ['jsx', 'coffee'],
});
return dgraph
.then(dg => dg.matchFilesByPattern('.*'))
.then(files => {
expect(files).toEqual(['/root/index.jsx', '/root/a.coffee']);
})
.then(() => getOrderedDependenciesAsJSON(dgraph, '/root/index.jsx'))
.then(deps => {
expect(deps).toEqual([
{
dependencies: ['a'],
id: 'index',
isAsset: false,
isJSON: false,
isPolyfill: false,
path: '/root/index.jsx',
resolution: undefined,
},
{
dependencies: [],
id: 'a',
isAsset: false,
isJSON: false,
isPolyfill: false,
path: '/root/a.coffee',
resolution: undefined,
},
]);
});
});
it('supports custom file extensions with relative paths', async () => {
const root = '/root';
setMockFileSystem({
root: {
'index.jsx': ['require("./a")'].join('\n'),
'a.coffee': [].join('\n'),
'X.js': '',
},
});
const dgraph = await DependencyGraph.load({
...defaults,
roots: [root],
sourceExts: ['jsx', 'coffee'],
});
const files = await dgraph.matchFilesByPattern('.*');
expect(files).toEqual(['/root/index.jsx', '/root/a.coffee']);
const deps = await getOrderedDependenciesAsJSON(
dgraph,
'/root/index.jsx',
);
expect(deps).toEqual([
{
dependencies: ['./a'],
id: '/root/index.jsx',
isAsset: false,
isJSON: false,
isPolyfill: false,
path: '/root/index.jsx',
resolution: undefined,
},
{
dependencies: [],
id: '/root/a.coffee',
isAsset: false,
isJSON: false,
isPolyfill: false,
path: '/root/a.coffee',
resolution: undefined,
},
]);
});
it('does not include extensions that are not specified explicitely', async () => {
const root = '/root';
setMockFileSystem({
root: {
'index.jsx': ['require("./a")'].join('\n'),
'a.coffee': [].join('\n'),
'X.js': '',
},
});
const dgraph = await DependencyGraph.load({
...defaults,
roots: [root],
});
const files = await dgraph.matchFilesByPattern('.*');
expect(files).toEqual(['/root/X.js']);
try {
await getOrderedDependenciesAsJSON(dgraph, '/root/index.jsx');
throw Error('should not reach this line');
} catch (error) {
expect(error.type).toEqual('UnableToResolveError');
}
});
});
describe('Progress updates', () => {
let dependencyGraph, onProgress;
function makeModule(id, dependencies = []) {
return (
`
/**
* @providesModule ${id}
*/\n` +
dependencies.map(d => `require(${JSON.stringify(d)});`).join('\n')
);
}
function getDependencies() {
return dependencyGraph.getDependencies({
entryPath: '/root/index.js',
onProgress,
options: emptyTransformOptions,
});
}
beforeEach(function() {
onProgress = jest.genMockFn();
setMockFileSystem({
root: {
'index.js': makeModule('index', ['a', 'b']),
'a.js': makeModule('a', ['c', 'd']),
'b.js': makeModule('b', ['d', 'e']),
'c.js': makeModule('c'),
'd.js': makeModule('d', ['f']),
'e.js': makeModule('e', ['f']),
'f.js': makeModule('f', ['g']),
'g.js': makeModule('g'),
},
});
const DependencyGraph = require('../DependencyGraph');
return DependencyGraph.load({
...defaults,
roots: ['/root'],
}).then(dg => {
dependencyGraph = dg;
});
});
it('calls back for each finished module', () => {
return getDependencies().then(() =>
expect(onProgress.mock.calls.length).toBe(8),
);
});
it('increases the number of finished modules in steps of one', () => {
return getDependencies().then(() => {
const increments = onProgress.mock.calls.map(([finished]) => finished);
expect(increments).toEqual([1, 2, 3, 4, 5, 6, 7, 8]);
});
});
it('adds the number of discovered modules to the number of total modules', () => {
return getDependencies().then(() => {
const increments = onProgress.mock.calls.map(([, total]) => total);
expect(increments).toEqual([3, 5, 6, 6, 7, 7, 8, 8]);
});
});
});
describe('Asset module dependencies', () => {
let DependencyGraph;
beforeEach(() => {
DependencyGraph = require('../DependencyGraph');
});
it('allows setting dependencies for asset modules', () => {
const assetDependencies = ['/root/apple.png', '/root/banana.png'];
setMockFileSystem({
root: {
'index.js': 'require("./a.png")',
'a.png': '',
'apple.png': '',
'banana.png': '',
},
});
DependencyGraph.load({
...defaults,
assetDependencies,
roots: ['/root'],
})
.then(dependencyGraph =>
dependencyGraph.getDependencies({
entryPath: '/root/index.js',
}),
)
.then(({dependencies}) => {
const [, assetModule] = dependencies;
return assetModule
.getDependencies()
.then(deps => expect(deps).toBe(assetDependencies));
});
});
});
describe('Deterministic order of dependencies', () => {
let callDeferreds, dependencyGraph, moduleReadDeferreds;
let moduleRead;
let DependencyGraph;
beforeEach(() => {
moduleRead = Module.prototype.read;
DependencyGraph = require('../DependencyGraph');
setMockFileSystem({
root: {
'index.js': `
require('./a');
require('./b');
`,
'a.js': `
require('./c');
require('./d');
`,
'b.js': `
require('./c');
require('./d');
`,
'c.js': 'require("./e");',
'd.js': '',
'e.js': 'require("./f");',
'f.js': 'require("./c");', // circular dependency
},
});
dependencyGraph = DependencyGraph.load({
...defaults,
roots: ['/root'],
});
moduleReadDeferreds = {};
callDeferreds = [defer(), defer()]; // [a.js, b.js]
Module.prototype.read = jest.genMockFn().mockImplementation(function() {
const returnValue = moduleRead.apply(this, arguments);
if (/\/[ab]\.js$/.test(this.path)) {
let deferred = moduleReadDeferreds[this.path];
if (!deferred) {
deferred = moduleReadDeferreds[this.path] = defer(returnValue);
const index = Number(this.path.endsWith('b.js')); // 0 or 1
callDeferreds[index].resolve();
}
return deferred.promise;
}
return returnValue;
});
});
afterEach(() => {
Module.prototype.read = moduleRead;
});
it('produces a deterministic tree if the "a" module resolves first', () => {
const dependenciesPromise = getOrderedDependenciesAsJSON(
dependencyGraph,
'index.js',
);
return Promise.all(callDeferreds.map(deferred => deferred.promise))
.then(() => {
const main = moduleReadDeferreds['/root/a.js'];
main.promise.then(() => {
moduleReadDeferreds['/root/b.js'].resolve();
});
main.resolve();
return dependenciesPromise;
})
.then(result => {
const names = result.map(({path: resultPath}) =>
resultPath.split('/').pop(),
);
expect(names).toEqual([
'index.js',
'a.js',
'c.js',
'e.js',
'f.js',
'd.js',
'b.js',
]);
});
});
it('produces a deterministic tree if the "b" module resolves first', () => {
const dependenciesPromise = getOrderedDependenciesAsJSON(
dependencyGraph,
'index.js',
);
return Promise.all(callDeferreds.map(deferred => deferred.promise))
.then(() => {
const main = moduleReadDeferreds['/root/b.js'];
main.promise.then(() => {
moduleReadDeferreds['/root/a.js'].resolve();
});
main.resolve();
return dependenciesPromise;
})
.then(result => {
const names = result.map(({path: resultPath}) =>
resultPath.split('/').pop(),
);
expect(names).toEqual([
'index.js',
'a.js',
'c.js',
'e.js',
'f.js',
'd.js',
'b.js',
]);
});
});
});
function defer(value) {
let resolve;
const promise = new Promise(r => {
resolve = r;
});
return {promise, resolve: () => resolve(value)};
}
function setMockFileSystem(object) {
return require('graceful-fs').__setMockFilesystem(object);
}
function triggerAndProcessWatchEvent(dgraphPromise, eventType, filename) {
return dgraphPromise.then(
dgraph =>
new Promise(resolve => {
dgraph.once('change', () => resolve());
triggerWatchEvent(eventType, filename);
}),
);
}
function triggerWatchEvent(eventType, filename) {
return require('graceful-fs').__triggerWatchEvent(eventType, filename);
}
});