Improve the resolution of relative paths

- resolve module relative path for component
 - prioritize coffeescript over json
This commit is contained in:
Pradnya Baviskar 2015-03-23 13:49:11 +05:30
parent d7c67c803a
commit 07d276977a
2 changed files with 350 additions and 72 deletions

View File

@ -8,6 +8,8 @@ var debug = require('debug')('loopback:boot:compiler');
var Module = require('module'); var Module = require('module');
var _ = require('lodash'); var _ = require('lodash');
var FILE_EXTENSION_JSON = '.json';
/** /**
* Gather all bootstrap-related configuration data and compile it into * Gather all bootstrap-related configuration data and compile it into
* a single object containing instruction for `boot.execute`. * a single object containing instruction for `boot.execute`.
@ -160,12 +162,11 @@ function findScripts(dir) {
} }
var filepath = path.resolve(path.join(dir, filename)); var filepath = path.resolve(path.join(dir, filename));
var ext = path.extname(filename);
var stats = fs.statSync(filepath); var stats = fs.statSync(filepath);
// only require files supported by require.extensions (.txt .md etc.) // only require files supported by require.extensions (.txt .md etc.)
if (stats.isFile()) { if (stats.isFile()) {
if (ext in require.extensions) if (isPreferredExtension(filename))
results.push(filepath); results.push(filepath);
else else
debug('Skipping file %s - unknown extension', filepath); debug('Skipping file %s - unknown extension', filepath);
@ -299,8 +300,8 @@ function findModelDefinitions(rootDir, sources) {
return registry; return registry;
} }
function resolveAppPath(rootDir, relativePath) { function resolveAppPath(rootDir, relativePath, resolveOptions) {
var resolvedPath = tryResolveAppPath(rootDir, relativePath); var resolvedPath = tryResolveAppPath(rootDir, relativePath, resolveOptions);
if (resolvedPath === undefined) { if (resolvedPath === undefined) {
var err = new Error('Cannot resolve path "' + relativePath + '"'); var err = new Error('Cannot resolve path "' + relativePath + '"');
err.code = 'PATH_NOT_FOUND'; err.code = 'PATH_NOT_FOUND';
@ -309,12 +310,27 @@ function resolveAppPath(rootDir, relativePath) {
return resolvedPath; return resolvedPath;
} }
function tryResolveAppPath(rootDir, relativePath) { function tryResolveAppPath(rootDir, relativePath, resolveOptions) {
var fullPath = path.resolve(rootDir, relativePath); var fullPath;
if (fs.existsSync(fullPath)) var start = relativePath.substring(0, 2);
/* In order to retain backward compatibility, while resolving
* component path, `resolveOptions` parameter is added where
* `resolveOptions.strict` = true,
* means retain backward compatibility when resolving the path and
* `resolveOptions.strict` = false,
* does not enforce any such restriction when resolving the path */
resolveOptions = resolveOptions || { strict: false };
if (relativePath[0] === '/') {
fullPath = relativePath;
} else if (start === './' || start === '..' || !resolveOptions.strict) {
fullPath = path.resolve(rootDir, relativePath);
}
if (fullPath && fs.existsSync(fullPath))
return fullPath; return fullPath;
var start = relativePath.substring(0, 2);
if (start !== './' && start !== '..') { if (start !== './' && start !== '..') {
// Handle module-relative path, e.g. `loopback/common/models` // Handle module-relative path, e.g. `loopback/common/models`
@ -339,6 +355,7 @@ function tryResolveAppPath(rootDir, relativePath) {
return fs.existsSync(candidate); return fs.existsSync(candidate);
}) })
[0]; [0];
if (fullPath) if (fullPath)
return fullPath; return fullPath;
} else { } else {
@ -356,27 +373,12 @@ function tryResolveAppPath(rootDir, relativePath) {
function loadModelDefinition(rootDir, jsonFile, allFiles) { function loadModelDefinition(rootDir, jsonFile, allFiles) {
var definition = require(jsonFile); var definition = require(jsonFile);
var basename = path.basename(jsonFile, path.extname(jsonFile));
// find a matching file with a supported extension like `.js` or `.coffee` // find a matching file with a supported extension like `.js` or `.coffee`
var base; var sourceFile = fixFileExtension(jsonFile, allFiles, true);
var ext;
var validFileType;
var sourceFile = allFiles
.filter(function(f) {
ext = path.extname(f);
base = path.basename(f, ext);
validFileType = (ext !== '.node') && (ext !== '.json') &&
((typeof require.extensions[ext]) === 'function');
return validFileType && (base === basename);
})[0];
try { if (sourceFile === undefined) {
sourceFile = path.join(path.dirname(jsonFile), sourceFile); debug('Model source code not found: %s', sourceFile);
sourceFile = require.resolve(sourceFile);
} catch (err) {
debug('Model source code not found: %s - %s', sourceFile, err.code || err);
sourceFile = undefined;
} }
debug('Found model "%s" - %s %s', definition.name, debug('Found model "%s" - %s %s', definition.name,
@ -456,7 +458,7 @@ function resolveMiddlewarePath(rootDir, middleware) {
} }
if (!fragment) { if (!fragment) {
resolved.sourceFile = resolveAppPath(rootDir, middlewarePath); resolved.sourceFile = resolveAppScriptPath(rootDir, middlewarePath);
return resolved; return resolved;
} }
@ -466,7 +468,7 @@ function resolveMiddlewarePath(rootDir, middleware) {
// function // function
var m = require(pathName); var m = require(pathName);
if (typeof m[fragment] === 'function') { if (typeof m[fragment] === 'function') {
resolved.sourceFile = resolveAppPath(rootDir, middlewarePath); resolved.sourceFile = resolveAppScriptPath(rootDir, middlewarePath);
return resolved; return resolved;
} }
@ -484,7 +486,7 @@ function resolveMiddlewarePath(rootDir, middleware) {
for (var ix in candidates) { for (var ix in candidates) {
try { try {
resolved.sourceFile = resolveAppPath(rootDir, candidates[ix]); resolved.sourceFile = resolveAppScriptPath(rootDir, candidates[ix]);
delete resolved.fragment; delete resolved.fragment;
return resolved; return resolved;
} }
@ -513,16 +515,8 @@ function buildComponentInstructions(rootDir, componentConfig) {
return Object.keys(componentConfig) return Object.keys(componentConfig)
.filter(function(name) { return !!componentConfig[name]; }) .filter(function(name) { return !!componentConfig[name]; })
.map(function(name) { .map(function(name) {
var sourceFile;
if (name.indexOf('./') === 0 || name.indexOf('../') === 0) {
// Relative path
sourceFile = path.resolve(rootDir, name);
} else {
sourceFile = require.resolve(name);
}
return { return {
sourceFile: sourceFile, sourceFile: resolveAppScriptPath(rootDir, name, {'strict': true}),
config: componentConfig[name] config: componentConfig[name]
}; };
}); });
@ -538,3 +532,56 @@ function resolveRelativePaths(relativePaths, appRootDir) {
} }
}); });
} }
function getExcludedExtensions() {
return {
'.json': '.json',
'.node': 'node'
};
}
function isPreferredExtension (filename) {
var includeExtensions = require.extensions;
var ext = path.extname(filename);
return (ext in includeExtensions) && !(ext in getExcludedExtensions());
}
function fixFileExtension (filepath, files, onlyScriptsExportingFunction) {
var results = [];
var otherFile;
/* Prefer coffee scripts over json */
if (isPreferredExtension(filepath)) return filepath;
var basename = path.basename(filepath, FILE_EXTENSION_JSON);
var sourceDir = path.dirname(filepath);
files.forEach(function(f) {
otherFile = path.resolve(sourceDir, f);
var stats = fs.statSync(otherFile);
if (stats.isFile()) {
var otherFileExtension = path.extname(f);
if (!(otherFileExtension in getExcludedExtensions()) &&
path.basename(f, otherFileExtension) == basename) {
if (!onlyScriptsExportingFunction)
results.push(otherFile);
else if (onlyScriptsExportingFunction &&
(typeof require.extensions[otherFileExtension]) === 'function') {
results.push(otherFile);
}
}
}
});
return (results.length > 0 ? results[0] : undefined);
}
function resolveAppScriptPath (rootDir, relativePath, resolveOptions) {
var resolvedPath = resolveAppPath (rootDir, relativePath, resolveOptions);
var sourceDir = path.dirname(resolvedPath);
var files = tryReadDir(sourceDir);
var fixedFile = fixFileExtension (resolvedPath, files, false);
return (fixedFile === undefined ? resolvedPath : fixedFile);
}

View File

@ -5,6 +5,9 @@ var expect = require('chai').expect;
var sandbox = require('./helpers/sandbox'); var sandbox = require('./helpers/sandbox');
var appdir = require('./helpers/appdir'); var appdir = require('./helpers/appdir');
// add coffee-script to require.extensions
require('coffee-script/register');
var SIMPLE_APP = path.join(__dirname, 'fixtures', 'simple-app'); var SIMPLE_APP = path.join(__dirname, 'fixtures', 'simple-app');
describe('compiler', function() { describe('compiler', function() {
@ -394,6 +397,63 @@ describe('compiler', function() {
expect(instructions.files.boot).to.eql([initJs]); expect(instructions.files.boot).to.eql([initJs]);
}); });
it('should resolve relative path in `bootDirs`', function() {
appdir.createConfigFilesSync();
var initJs = appdir.writeFileSync('custom-boot/init.js',
'module.exports = function(app) { app.fnCalled = true; };');
var instructions = boot.compile({
appRootDir: appdir.PATH,
bootDirs:['./custom-boot']
});
expect(instructions.files.boot).to.eql([initJs]);
});
it('should resolve non-relative path in `bootDirs`', function() {
appdir.createConfigFilesSync();
var initJs = appdir.writeFileSync('custom-boot/init.js', '');
var instructions = boot.compile({
appRootDir: appdir.PATH,
bootDirs:['custom-boot']
});
expect(instructions.files.boot).to.eql([initJs]);
});
it('ignores index.js in `bootDirs`', function() {
appdir.createConfigFilesSync();
appdir.writeFileSync('custom-boot/index.js', '');
var instructions = boot.compile({
appRootDir: appdir.PATH,
bootDirs:['./custom-boot']
});
expect(instructions.files.boot).to.have.length(0);
});
it('prefers coffeescript over json in `appRootDir/bootDir`', function() {
appdir.createConfigFilesSync();
var coffee = appdir.writeFileSync('./custom-boot/init.coffee', '');
appdir.writeFileSync('./custom-boot/init.json', {});
var instructions = boot.compile({
appRootDir: appdir.PATH,
bootDirs: ['./custom-boot']
});
expect(instructions.files.boot).to.eql([coffee]);
});
it('prefers coffeescript over json in `bootDir` non-relative path',
function() {
appdir.createConfigFilesSync();
var coffee = appdir.writeFileSync('custom-boot/init.coffee',
'');
appdir.writeFileSync('custom-boot/init.json', '');
var instructions = boot.compile({
appRootDir: appdir.PATH,
bootDirs: ['custom-boot']
});
expect(instructions.files.boot).to.eql([coffee]);
});
it('supports `bootScripts` option', function() { it('supports `bootScripts` option', function() {
appdir.createConfigFilesSync(); appdir.createConfigFilesSync();
var initJs = appdir.writeFileSync('custom-boot/init.js', var initJs = appdir.writeFileSync('custom-boot/init.js',
@ -428,17 +488,6 @@ describe('compiler', function() {
expect(instructions.files.boot).to.eql([initJs]); expect(instructions.files.boot).to.eql([initJs]);
}); });
it('should resolve relative path in `bootDirs`', function() {
appdir.createConfigFilesSync();
var initJs = appdir.writeFileSync('custom-boot/init.js',
'module.exports = function(app) { app.fnCalled = true; };');
var instructions = boot.compile({
appRootDir: appdir.PATH,
bootDirs:['./custom-boot']
});
expect(instructions.files.boot).to.eql([initJs]);
});
it('should resolve non-relative path in `bootScripts`', function() { it('should resolve non-relative path in `bootScripts`', function() {
appdir.createConfigFilesSync(); appdir.createConfigFilesSync();
var initJs = appdir.writeFileSync('custom-boot/init.js', ''); var initJs = appdir.writeFileSync('custom-boot/init.js', '');
@ -449,16 +498,6 @@ describe('compiler', function() {
expect(instructions.files.boot).to.eql([initJs]); expect(instructions.files.boot).to.eql([initJs]);
}); });
it('should resolve non-relative path in `bootDirs`', function() {
appdir.createConfigFilesSync();
var initJs = appdir.writeFileSync('custom-boot/init.js', '');
var instructions = boot.compile({
appRootDir: appdir.PATH,
bootDirs:['custom-boot']
});
expect(instructions.files.boot).to.eql([initJs]);
});
it('resolves missing extensions in `bootScripts`', function() { it('resolves missing extensions in `bootScripts`', function() {
appdir.createConfigFilesSync(); appdir.createConfigFilesSync();
var initJs = appdir.writeFileSync('custom-boot/init.js', ''); var initJs = appdir.writeFileSync('custom-boot/init.js', '');
@ -481,16 +520,6 @@ describe('compiler', function() {
expect(instructions.files.boot).to.eql([initJs]); expect(instructions.files.boot).to.eql([initJs]);
}); });
it('ignores index.js in `bootDirs`', function() {
appdir.createConfigFilesSync();
appdir.writeFileSync('custom-boot/index.js', '');
var instructions = boot.compile({
appRootDir: appdir.PATH,
bootDirs:['./custom-boot']
});
expect(instructions.files.boot).to.have.length(0);
});
it('resolves module relative path for `bootScripts`', function() { it('resolves module relative path for `bootScripts`', function() {
appdir.createConfigFilesSync(); appdir.createConfigFilesSync();
var initJs = appdir.writeFileSync('node_modules/custom-boot/init.js', ''); var initJs = appdir.writeFileSync('node_modules/custom-boot/init.js', '');
@ -564,9 +593,6 @@ describe('compiler', function() {
}); });
it('loads coffeescript models from `./models`', function() { it('loads coffeescript models from `./models`', function() {
// add coffee-script to require.extensions
require('coffee-script/register');
appdir.createConfigFilesSync({}, {}, { appdir.createConfigFilesSync({}, {}, {
Car: { dataSource: 'db' } Car: { dataSource: 'db' }
}); });
@ -662,6 +688,75 @@ describe('compiler', function() {
}); });
}); });
it('resolves relative path in `modelSources` option', function() {
appdir.createConfigFilesSync({}, {}, {
Car: { dataSource: 'db' }
});
appdir.writeConfigFileSync('custom-models/car.json', { name: 'Car' });
var appJS = appdir.writeFileSync('custom-models/car.js', '');
var instructions = boot.compile({
appRootDir: appdir.PATH,
modelSources: ['./custom-models']
});
expect(instructions.models).to.have.length(1);
expect(instructions.models[0].sourceFile).to.equal(appJS);
});
it('resolves module relative path in `modelSources` option', function() {
appdir.createConfigFilesSync({}, {}, {
Car: { dataSource: 'db' }
});
appdir.writeConfigFileSync('node_modules/custom-models/car.json',
{ name: 'Car' });
var appJS = appdir.writeFileSync('node_modules/custom-models/car.js', '');
var instructions = boot.compile({
appRootDir: appdir.PATH,
modelSources: ['custom-models']
});
expect(instructions.models).to.have.length(1);
expect(instructions.models[0].sourceFile).to.equal(appJS);
});
it('resolves relative path in `sources` option in `model-config.json`',
function() {
appdir.createConfigFilesSync({}, {}, {
_meta: {
sources: ['./custom-models']
},
Car: { dataSource: 'db' }
});
appdir.writeConfigFileSync('custom-models/car.json', { name: 'Car' });
var appJS = appdir.writeFileSync('custom-models/car.js', '');
var instructions = boot.compile(appdir.PATH);
expect(instructions.models).to.have.length(1);
expect(instructions.models[0].sourceFile).to.equal(appJS);
});
it('resolves module relative path in `sources` option in model-config.json',
function() {
appdir.createConfigFilesSync({}, {}, {
_meta: {
sources: ['custom-models']
},
Car: { dataSource: 'db' }
});
appdir.writeConfigFileSync('node_modules/custom-models/car.json',
{ name: 'Car' });
var appJS = appdir.writeFileSync('node_modules/custom-models/car.js', '');
var instructions = boot.compile(appdir.PATH);
expect(instructions.models).to.have.length(1);
expect(instructions.models[0].sourceFile).to.equal(appJS);
});
it('handles model definitions with no code', function() { it('handles model definitions with no code', function() {
appdir.createConfigFilesSync({}, {}, { appdir.createConfigFilesSync({}, {}, {
Car: { dataSource: 'db' } Car: { dataSource: 'db' }
@ -1096,6 +1191,73 @@ describe('compiler', function() {
'sourceFile', moduleJS); 'sourceFile', moduleJS);
}); });
it('loads middleware from coffeescript in appRootdir', function() {
var coffee = appdir.writeFileSync('my-middleware.coffee', '');
appdir.writeConfigFileSync('middleware.json', {
'routes': {
'./my-middleware': {}
}
});
var instructions = boot.compile(appdir.PATH);
expect(instructions.middleware.middleware[0]).have.property(
'sourceFile', coffee);
});
it('loads coffeescript from middleware under node_modules',
function() {
var file = appdir.writeFileSync('node_modules/my-middleware/index.coffee',
'');
appdir.writeFileSync('node_modules/my-middleware/index.json', '');
appdir.writeConfigFileSync('middleware.json', {
'routes': {
'my-middleware': {}
}
});
var instructions = boot.compile(appdir.PATH);
expect(instructions.middleware.middleware).to.have.length(1);
expect(instructions.middleware.middleware[0]).have.property(
'sourceFile', file);
});
it('prefers coffeescript over json for relative middleware path',
function() {
var coffee = appdir.writeFileSync('my-middleware.coffee', '');
appdir.writeFileSync('my-middleware.json', '');
appdir.writeConfigFileSync('middleware.json', {
'routes': {
'./my-middleware': {}
}
});
var instructions = boot.compile(appdir.PATH);
expect(instructions.middleware.middleware).to.have.length(1);
expect(instructions.middleware.middleware[0]).have.property(
'sourceFile', coffee);
});
it('prefers coffeescript over json for module relative middleware path',
function() {
var coffee = appdir.writeFileSync('node_modules/my-middleware.coffee',
'');
appdir.writeFileSync('node_modules/my-middleware.json', '');
appdir.writeConfigFileSync('middleware.json', {
'routes': {
'my-middleware': {}
}
});
var instructions = boot.compile(appdir.PATH);
expect(instructions.middleware.middleware).to.have.length(1);
expect(instructions.middleware.middleware[0]).have.property(
'sourceFile', coffee);
});
describe('config with relative paths in params', function() { describe('config with relative paths in params', function() {
var RELATIVE_PATH_PARAMS = [ var RELATIVE_PATH_PARAMS = [
'$!./here', '$!./here',
@ -1195,6 +1357,75 @@ describe('compiler', function() {
} }
}); });
}); });
it('loads component relative to appRootDir', function() {
appdir.writeConfigFileSync('./component-config.json', {
'./index': { }
});
var appJS = appdir.writeConfigFileSync('index.js', '');
var instructions = boot.compile(appdir.PATH);
expect(instructions.components[0]).have.property(
'sourceFile', appJS
);
});
it('loads component relative to node modules', function() {
appdir.writeConfigFileSync('component-config.json', {
'mycomponent': { }
});
var js = appdir.writeConfigFileSync('node_modules/mycomponent/index.js',
'');
var instructions = boot.compile(appdir.PATH);
expect(instructions.components[0]).have.property(
'sourceFile', js
);
});
it('retains backward compatibility for non-relative path in `appRootDir`',
function() {
appdir.writeConfigFileSync('component-config.json', {
'my-component/component.js': { }
});
appdir.writeConfigFileSync('./my-component/component.js', '');
expect(function() { boot.compile(appdir.PATH); })
.to.throw('Cannot resolve path \"my-component/component.js\"');
});
it('prefers coffeescript over json for relative path component',
function() {
appdir.writeConfigFileSync('component-config.json', {
'./component': { }
});
var coffee = appdir.writeFileSync('component.coffee', '');
appdir.writeFileSync('component.json', '');
var instructions = boot.compile(appdir.PATH);
expect(instructions.components).to.have.length(1);
expect(instructions.components[0]).have.property(
'sourceFile', coffee);
});
it('prefers coffeescript over json for module relative component path',
function() {
appdir.writeConfigFileSync('component-config.json', {
'component': { }
});
var coffee = appdir.writeFileSync('node_modules/component.coffee', '');
appdir.writeFileSync('node_modules/component.json', '');
var instructions = boot.compile(appdir.PATH);
expect(instructions.components).to.have.length(1);
expect(instructions.components[0]).have.property(
'sourceFile', coffee);
});
}); });
}); });