webpack/lib/ContextModuleFactory.js

250 lines
6.2 KiB
JavaScript
Raw Normal View History

2013-01-31 01:49:25 +08:00
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
2018-07-30 23:08:51 +08:00
2017-05-11 03:36:20 +08:00
"use strict";
2018-02-11 12:27:09 +08:00
const asyncLib = require("neo-async");
2017-05-11 03:36:20 +08:00
const path = require("path");
2018-06-26 14:27:44 +08:00
const { AsyncSeriesWaterfallHook, SyncWaterfallHook } = require("tapable");
2017-05-11 03:36:20 +08:00
const ContextModule = require("./ContextModule");
const ContextElementDependency = require("./dependencies/ContextElementDependency");
/** @typedef {import("./Module")} Module */
const EMPTY_RESOLVE_OPTIONS = {};
2018-06-26 14:27:44 +08:00
module.exports = class ContextModuleFactory {
constructor(resolverFactory) {
2018-07-30 20:25:40 +08:00
this.hooks = Object.freeze({
/** @type {AsyncSeriesWaterfallHook<TODO>} */
2017-11-28 16:54:24 +08:00
beforeResolve: new AsyncSeriesWaterfallHook(["data"]),
/** @type {AsyncSeriesWaterfallHook<TODO>} */
2017-11-28 16:54:24 +08:00
afterResolve: new AsyncSeriesWaterfallHook(["data"]),
/** @type {SyncWaterfallHook<string[]>} */
2017-11-28 16:54:24 +08:00
contextModuleFiles: new SyncWaterfallHook(["files"]),
/** @type {SyncWaterfallHook<TODO[]>} */
2017-11-28 16:54:24 +08:00
alternatives: new AsyncSeriesWaterfallHook(["modules"])
2018-07-30 20:25:40 +08:00
});
this.resolverFactory = resolverFactory;
2017-05-11 03:36:20 +08:00
}
create(data, callback) {
const context = data.context;
const dependencies = data.dependencies;
const resolveOptions = data.resolveOptions;
2017-05-11 03:36:20 +08:00
const dependency = dependencies[0];
2018-02-25 09:00:20 +08:00
this.hooks.beforeResolve.callAsync(
Object.assign(
{
context: context,
dependencies: dependencies,
resolveOptions
2017-05-11 03:36:20 +08:00
},
2018-02-25 09:00:20 +08:00
dependency.options
),
(err, beforeResolveResult) => {
if (err) return callback(err);
// Ignored
if (!beforeResolveResult) return callback();
const context = beforeResolveResult.context;
const request = beforeResolveResult.request;
const resolveOptions = beforeResolveResult.resolveOptions;
let loaders,
resource,
loadersPrefix = "";
const idx = request.lastIndexOf("!");
if (idx >= 0) {
let loadersRequest = request.substr(0, idx + 1);
2018-02-25 09:00:20 +08:00
let i;
for (
i = 0;
i < loadersRequest.length && loadersRequest[i] === "!";
i++
) {
2018-02-25 09:00:20 +08:00
loadersPrefix += "!";
}
loadersRequest = loadersRequest
2018-02-25 09:00:20 +08:00
.substr(i)
.replace(/!+$/, "")
.replace(/!!+/g, "!");
if (loadersRequest === "") {
loaders = [];
} else {
loaders = loadersRequest.split("!");
}
2018-02-25 09:00:20 +08:00
resource = request.substr(idx + 1);
} else {
loaders = [];
resource = request;
2017-05-11 03:36:20 +08:00
}
2018-02-25 09:00:20 +08:00
const contextResolver = this.resolverFactory.get(
"context",
resolveOptions || EMPTY_RESOLVE_OPTIONS
);
const loaderResolver = this.resolverFactory.get(
"loader",
EMPTY_RESOLVE_OPTIONS
);
asyncLib.parallel(
[
callback => {
contextResolver.resolve(
{},
context,
resource,
{},
(err, result) => {
if (err) return callback(err);
callback(null, result);
}
);
},
callback => {
asyncLib.map(
loaders,
(loader, callback) => {
loaderResolver.resolve(
{},
context,
loader,
{},
(err, result) => {
if (err) return callback(err);
callback(null, result);
}
);
},
callback
);
}
],
(err, result) => {
if (err) return callback(err);
this.hooks.afterResolve.callAsync(
Object.assign(
{
addon:
loadersPrefix +
result[1].join("!") +
(result[1].length > 0 ? "!" : ""),
resource: result[0],
resolveDependencies: this.resolveDependencies.bind(this)
},
beforeResolveResult
),
(err, result) => {
if (err) return callback(err);
// Ignored
if (!result) return callback();
return callback(
null,
new ContextModule(result.resolveDependencies, result)
);
}
);
}
);
}
);
2017-05-11 03:36:20 +08:00
}
2013-02-04 19:34:20 +08:00
2017-10-14 07:51:01 +08:00
resolveDependencies(fs, options, callback) {
const cmf = this;
let resource = options.resource;
let resourceQuery = options.resourceQuery;
let recursive = options.recursive;
let regExp = options.regExp;
let include = options.include;
let exclude = options.exclude;
2018-02-25 09:00:20 +08:00
if (!regExp || !resource) return callback(null, []);
2017-11-08 18:32:05 +08:00
const addDirectory = (directory, callback) => {
2017-08-11 13:52:25 +08:00
fs.readdir(directory, (err, files) => {
2018-02-25 09:00:20 +08:00
if (err) return callback(err);
2017-11-28 16:54:24 +08:00
files = cmf.hooks.contextModuleFiles.call(files);
2018-02-25 09:00:20 +08:00
if (!files || files.length === 0) return callback(null, []);
asyncLib.map(
files.filter(p => p.indexOf(".") !== 0),
2018-02-26 10:43:37 +08:00
(segment, callback) => {
const subResource = path.join(directory, segment);
2018-02-25 09:00:20 +08:00
if (!exclude || !subResource.match(exclude)) {
fs.stat(subResource, (err, stat) => {
if (err) {
if (err.code === "ENOENT") {
// ENOENT is ok here because the file may have been deleted between
// the readdir and stat calls.
return callback();
} else {
return callback(err);
}
2017-10-14 07:51:01 +08:00
}
2013-02-04 19:34:20 +08:00
2018-02-25 09:00:20 +08:00
if (stat.isDirectory()) {
if (!recursive) return callback();
addDirectory.call(this, subResource, callback);
} else if (
stat.isFile() &&
(!include || subResource.match(include))
) {
const obj = {
context: resource,
request:
"." +
subResource.substr(resource.length).replace(/\\/g, "/")
};
this.hooks.alternatives.callAsync(
[obj],
(err, alternatives) => {
if (err) return callback(err);
alternatives = alternatives
.filter(obj => regExp.test(obj.request))
.map(obj => {
const dep = new ContextElementDependency(
obj.request + resourceQuery,
obj.request
);
dep.optional = true;
return dep;
});
callback(null, alternatives);
}
);
} else {
callback();
}
2018-02-25 09:00:20 +08:00
});
} else {
callback();
}
2018-02-25 09:00:20 +08:00
},
(err, result) => {
if (err) return callback(err);
if (!result) return callback(null, []);
callback(
null,
result.filter(Boolean).reduce((a, i) => a.concat(i), [])
);
}
);
2017-08-11 13:52:25 +08:00
});
2017-11-08 18:32:05 +08:00
};
addDirectory(resource, callback);
2017-05-11 03:36:20 +08:00
}
2013-01-31 01:49:25 +08:00
};