201 lines
6.3 KiB
JavaScript
201 lines
6.3 KiB
JavaScript
/*
|
|
MIT License http://www.opensource.org/licenses/mit-license.php
|
|
Author Tobias Koppers @sokra
|
|
*/
|
|
var Tapable = require("tapable");
|
|
var createInnerCallback = require("./createInnerCallback");
|
|
|
|
function Resolver(fileSystem) {
|
|
Tapable.call(this);
|
|
this.fileSystem = fileSystem;
|
|
}
|
|
module.exports = Resolver;
|
|
|
|
Resolver.prototype = Object.create(Tapable.prototype);
|
|
|
|
Resolver.prototype.resolveSync = function resolveSync(context, request) {
|
|
var err, result, sync = false;
|
|
this.resolve(context, request, function(e, r) {
|
|
err = e;
|
|
result = r;
|
|
sync = true;
|
|
});
|
|
if(!sync) throw new Error("Cannot 'resolveSync' because the fileSystem is not sync. Use 'resolve'!");
|
|
if(err) throw err;
|
|
return result;
|
|
};
|
|
|
|
Resolver.prototype.resolve = function resolve(context, request, callback) {
|
|
if(typeof request === "string") request = this.parse(request);
|
|
this.applyPlugins("resolve", context, request);
|
|
var obj = {
|
|
path: context,
|
|
request: request.path,
|
|
query: request.query,
|
|
directory: request.directory
|
|
};
|
|
function onResolved(err, result) {
|
|
if(err) return callback(err);
|
|
return callback(null, result.path === false ? false : result.path + (result.query || ""));
|
|
}
|
|
onResolved.log = callback.log;
|
|
onResolved.missing = callback.missing;
|
|
if(request.module) return this.doResolve("module", obj, onResolved);
|
|
if(request.directory) return this.doResolve("directory", obj, onResolved);
|
|
return this.doResolve(["file", "directory"], obj, onResolved);
|
|
};
|
|
|
|
Resolver.prototype.doResolve = function doResolve(types, request, callback, noError) {
|
|
if(!Array.isArray(types)) types = [types];
|
|
var stackLine = types.join(" or ") + ": (" + request.path + ") " +
|
|
(request.request || "") + (request.query || "") +
|
|
(request.directory ? " directory" : "") +
|
|
(request.module ? " module" : "");
|
|
var newStack = [stackLine];
|
|
if(callback.stack) {
|
|
newStack = callback.stack.concat(newStack);
|
|
if(callback.stack.indexOf(stackLine) >= 0) {
|
|
// Prevent recursion
|
|
var recursionError = new Error("Recursion in resolving\nStack:\n " + newStack.join("\n "));
|
|
recursionError.recursion = true;
|
|
if(callback.log) callback.log("abort resolving because of recursion");
|
|
return callback(recursionError);
|
|
}
|
|
}
|
|
this.applyPlugins("resolve-step", types, request);
|
|
var localMissing = [];
|
|
var missing = callback.missing ? {
|
|
push: function(item) {
|
|
callback.missing.push(item);
|
|
localMissing.push(item);
|
|
}
|
|
} : localMissing;
|
|
var log = [];
|
|
function writeLog(msg) {
|
|
log.push(msg);
|
|
}
|
|
function logAsString() {
|
|
return log.join("\n");
|
|
}
|
|
var currentRequestString = request.request ? request.request + " in " + request.path : request.path;
|
|
if(types.length == 1 && !noError) {
|
|
// If only one type, we can pass the error.
|
|
return this.applyPluginsParallelBailResult(types[0], request, createInnerCallback(function innerCallback(err, result) {
|
|
if(callback.log) {
|
|
for(var i = 0; i < log.length; i++)
|
|
callback.log(log[i]);
|
|
}
|
|
if(err) return callback(err);
|
|
if(result) return callback(null, result);
|
|
if(types[0] === "result") return callback(null, request);
|
|
var error = new Error("Cannot resolve " + types[0] + " '" + request.request + "' in " + request.path);
|
|
error.details = logAsString();
|
|
error.missing = localMissing;
|
|
return callback(error);
|
|
}, {
|
|
log: writeLog,
|
|
missing: missing,
|
|
stack: newStack
|
|
}, "resolve " + types[0] + " " + currentRequestString));
|
|
}
|
|
// For multiple type we list the errors in the details although some of them are not important
|
|
this.forEachBail(types, function(type, callback) {
|
|
this.applyPluginsParallelBailResult(type, request, createInnerCallback(function(err, result) {
|
|
if(!err && result) return callback(result);
|
|
if (err) {
|
|
(err.message || "").split("\n").forEach(function(line) {
|
|
log.push(" " + line);
|
|
});
|
|
}
|
|
callback();
|
|
}, {
|
|
log: writeLog,
|
|
missing: missing,
|
|
stack: newStack
|
|
}, "resolve " + type));
|
|
}.bind(this), function(result) {
|
|
if(callback.log) {
|
|
callback.log("resolve '" + types.join("' or '") + "' " + currentRequestString);
|
|
for(var i = 0; i < log.length; i++)
|
|
callback.log(" " + log[i]);
|
|
}
|
|
if(noError && !result) return callback();
|
|
if(result) return callback(null, result);
|
|
var error = new Error("Cannot resolve '" + types.join("' or '") + "' " + currentRequestString);
|
|
error.details = logAsString();
|
|
error.missing = localMissing;
|
|
return callback(error);
|
|
});
|
|
};
|
|
|
|
Resolver.prototype.parse = function parse(identifier) {
|
|
if(identifier === "") return null;
|
|
var part = {
|
|
path: null,
|
|
query: null,
|
|
module: false,
|
|
directory: false,
|
|
file: false
|
|
};
|
|
var idxQuery = identifier.indexOf("?");
|
|
if(idxQuery == 0) {
|
|
part.query = identifier;
|
|
} else if(idxQuery > 0) {
|
|
part.path = identifier.slice(0, idxQuery);
|
|
part.query = identifier.slice(idxQuery);
|
|
} else {
|
|
part.path = identifier;
|
|
}
|
|
if(part.path) {
|
|
part.module = this.isModule(part.path);
|
|
if(part.directory = this.isDirectory(part.path)) {
|
|
part.path = part.path.substr(0, part.path.length - 1);
|
|
}
|
|
}
|
|
return part;
|
|
};
|
|
|
|
var notModuleRegExp = /^\.$|^\.[\\\/]|^\.\.$|^\.\.[\/\\]|^\/|^[A-Z]:[\\\/]/i;
|
|
Resolver.prototype.isModule = function isModule(path) {
|
|
return !notModuleRegExp.test(path);
|
|
};
|
|
|
|
var directoryRegExp = /[\/\\]$/i;
|
|
Resolver.prototype.isDirectory = function isDirectory(path) {
|
|
return directoryRegExp.test(path);
|
|
};
|
|
|
|
Resolver.prototype.join = require("memory-fs/lib/join");
|
|
|
|
Resolver.prototype.normalize = require("memory-fs/lib/normalize");
|
|
|
|
Resolver.prototype.forEachBail = function(array, iterator, callback) {
|
|
if(array.length == 0) return callback();
|
|
var currentPos = array.length;
|
|
var currentError, currentResult;
|
|
var done = [];
|
|
for(var i = 0; i < array.length; i++) {
|
|
var itCb = (function(i) {
|
|
return function() {
|
|
if(i >= currentPos) return; // ignore
|
|
var args = Array.prototype.slice.call(arguments);
|
|
done.push(i);
|
|
if(args.length > 0) {
|
|
currentPos = i + 1;
|
|
done = done.filter(function(item) {
|
|
return item <= i;
|
|
});
|
|
currentResult = args;
|
|
}
|
|
if(done.length == currentPos) {
|
|
callback.apply(null, currentResult);
|
|
currentPos = 0;
|
|
}
|
|
};
|
|
}(i));
|
|
iterator(array[i], itCb);
|
|
if(currentPos == 0) break;
|
|
}
|
|
};
|
|
|