2013-01-31 01:49:25 +08:00
|
|
|
/*
|
|
|
|
MIT License http://www.opensource.org/licenses/mit-license.php
|
|
|
|
Author Tobias Koppers @sokra
|
|
|
|
*/
|
2017-02-18 17:12:09 +08:00
|
|
|
"use strict";
|
|
|
|
const path = require("path");
|
2017-10-16 18:16:07 +08:00
|
|
|
const util = require("util");
|
2017-02-18 17:12:09 +08:00
|
|
|
const Module = require("./Module");
|
|
|
|
const OriginalSource = require("webpack-sources").OriginalSource;
|
|
|
|
const RawSource = require("webpack-sources").RawSource;
|
|
|
|
const AsyncDependenciesBlock = require("./AsyncDependenciesBlock");
|
2017-05-05 00:37:25 +08:00
|
|
|
const DepBlockHelpers = require("./dependencies/DepBlockHelpers");
|
|
|
|
const Template = require("./Template");
|
2013-01-31 01:49:25 +08:00
|
|
|
|
2017-02-18 17:12:09 +08:00
|
|
|
class ContextModule extends Module {
|
2017-10-16 20:49:51 +08:00
|
|
|
// type ContextMode = "sync" | "eager" | "weak" | "async-weak" | "lazy" | "lazy-once"
|
|
|
|
// type ContextOptions = { resource: string, recursive: boolean, regExp: RegExp, addon?: string, mode?: ContextMode, chunkName?: string, include?: RegExp, exclude?: RegExp }
|
|
|
|
// resolveDependencies: (fs: FS, options: ContextOptions, (err: Error?, dependencies: Dependency[]) => void) => void
|
|
|
|
// options: ContextOptions
|
2017-10-14 05:31:15 +08:00
|
|
|
constructor(resolveDependencies, options) {
|
2017-10-30 20:56:57 +08:00
|
|
|
super("javascript/dynamic");
|
2017-11-06 20:02:35 +08:00
|
|
|
|
|
|
|
// Info from Factory
|
2017-02-18 17:12:09 +08:00
|
|
|
this.resolveDependencies = resolveDependencies;
|
2017-11-16 14:20:50 +08:00
|
|
|
let resource, resourceQuery;
|
|
|
|
const queryIdx = options.resource.indexOf("?");
|
|
|
|
if(queryIdx >= 0) {
|
|
|
|
resource = options.resource.substr(0, queryIdx);
|
|
|
|
resourceQuery = options.resource.substr(queryIdx);
|
|
|
|
} else {
|
|
|
|
resource = options.resource;
|
|
|
|
resourceQuery = "";
|
|
|
|
}
|
|
|
|
this.options = Object.assign({}, options, {
|
|
|
|
resource: resource,
|
|
|
|
resourceQuery: resourceQuery
|
|
|
|
});
|
|
|
|
this.context = this.options.resource;
|
2017-11-17 21:26:23 +08:00
|
|
|
if(options.resolveOptions !== undefined)
|
|
|
|
this.resolveOptions = options.resolveOptions;
|
2017-11-06 20:02:35 +08:00
|
|
|
|
|
|
|
// Info from Build
|
|
|
|
this.builtTime = undefined;
|
2017-11-06 23:41:26 +08:00
|
|
|
this.contextDependencies = new Set([this.context]);
|
2017-11-06 20:02:35 +08:00
|
|
|
|
2017-10-16 20:49:51 +08:00
|
|
|
if(typeof options.mode !== "string")
|
|
|
|
throw new Error("options.mode is a required option");
|
2017-02-18 17:12:09 +08:00
|
|
|
}
|
2013-01-31 01:49:25 +08:00
|
|
|
|
2017-02-18 17:26:47 +08:00
|
|
|
prettyRegExp(regexString) {
|
|
|
|
// remove the "/" at the front and the beginning
|
|
|
|
// "/foo/" -> "foo"
|
|
|
|
return regexString.substring(1, regexString.length - 1);
|
2017-02-18 17:12:09 +08:00
|
|
|
}
|
2013-01-31 01:49:25 +08:00
|
|
|
|
2017-02-18 17:15:29 +08:00
|
|
|
contextify(context, request) {
|
2017-02-23 18:52:11 +08:00
|
|
|
return request.split("!").map(subrequest => {
|
2017-02-18 17:15:29 +08:00
|
|
|
let rp = path.relative(context, subrequest);
|
2017-02-18 17:12:09 +08:00
|
|
|
if(path.sep === "\\")
|
|
|
|
rp = rp.replace(/\\/g, "/");
|
|
|
|
if(rp.indexOf("../") !== 0)
|
|
|
|
rp = "./" + rp;
|
|
|
|
return rp;
|
|
|
|
}).join("!");
|
|
|
|
}
|
2013-01-31 01:49:25 +08:00
|
|
|
|
2017-02-18 17:29:21 +08:00
|
|
|
identifier() {
|
|
|
|
let identifier = this.context;
|
2017-10-16 20:49:51 +08:00
|
|
|
if(this.options.mode)
|
|
|
|
identifier += ` ${this.options.mode}`;
|
2017-10-14 05:31:15 +08:00
|
|
|
if(!this.options.recursive)
|
2017-02-18 17:29:21 +08:00
|
|
|
identifier += " nonrecursive";
|
2017-10-14 05:31:15 +08:00
|
|
|
if(this.options.addon)
|
|
|
|
identifier += ` ${this.options.addon}`;
|
|
|
|
if(this.options.regExp)
|
|
|
|
identifier += ` ${this.options.regExp}`;
|
|
|
|
if(this.options.include)
|
|
|
|
identifier += ` include: ${this.options.include}`;
|
|
|
|
if(this.options.exclude)
|
|
|
|
identifier += ` exclude: ${this.options.exclude}`;
|
2017-02-18 17:29:21 +08:00
|
|
|
|
|
|
|
return identifier;
|
|
|
|
}
|
|
|
|
|
|
|
|
readableIdentifier(requestShortener) {
|
|
|
|
let identifier = requestShortener.shorten(this.context);
|
2017-10-16 20:49:51 +08:00
|
|
|
if(this.options.mode)
|
|
|
|
identifier += ` ${this.options.mode}`;
|
2017-10-14 05:31:15 +08:00
|
|
|
if(!this.options.recursive)
|
2017-02-18 17:29:21 +08:00
|
|
|
identifier += " nonrecursive";
|
2017-10-14 05:31:15 +08:00
|
|
|
if(this.options.addon)
|
|
|
|
identifier += ` ${requestShortener.shorten(this.options.addon)}`;
|
|
|
|
if(this.options.regExp)
|
|
|
|
identifier += ` ${this.prettyRegExp(this.options.regExp + "")}`;
|
|
|
|
if(this.options.include)
|
|
|
|
identifier += ` include: ${this.prettyRegExp(this.options.include + "")}`;
|
|
|
|
if(this.options.exclude)
|
|
|
|
identifier += ` exclude: ${this.prettyRegExp(this.options.exclude + "")}`;
|
2017-02-18 17:29:21 +08:00
|
|
|
|
|
|
|
return identifier;
|
|
|
|
}
|
|
|
|
|
2017-02-18 17:12:09 +08:00
|
|
|
libIdent(options) {
|
2017-02-18 17:29:21 +08:00
|
|
|
let identifier = this.contextify(options.context, this.context);
|
2017-10-16 20:49:51 +08:00
|
|
|
if(this.options.mode)
|
|
|
|
identifier += ` ${this.options.mode}`;
|
2017-10-14 05:31:15 +08:00
|
|
|
if(this.options.recursive)
|
2017-02-18 17:29:21 +08:00
|
|
|
identifier += " recursive";
|
2017-10-14 05:31:15 +08:00
|
|
|
if(this.options.addon)
|
|
|
|
identifier += ` ${this.contextify(options.context, this.options.addon)}`;
|
|
|
|
if(this.options.regExp)
|
|
|
|
identifier += ` ${this.prettyRegExp(this.options.regExp + "")}`;
|
|
|
|
if(this.options.include)
|
|
|
|
identifier += ` include: ${this.prettyRegExp(this.options.include + "")}`;
|
|
|
|
if(this.options.exclude)
|
|
|
|
identifier += ` exclude: ${this.prettyRegExp(this.options.exclude + "")}`;
|
2017-02-18 17:29:21 +08:00
|
|
|
|
|
|
|
return identifier;
|
2017-02-18 17:12:09 +08:00
|
|
|
}
|
2016-12-14 19:03:24 +08:00
|
|
|
|
2017-02-18 17:12:09 +08:00
|
|
|
needRebuild(fileTimestamps, contextTimestamps) {
|
|
|
|
const ts = contextTimestamps[this.context];
|
2017-02-18 18:09:18 +08:00
|
|
|
if(!ts) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-02-18 17:12:09 +08:00
|
|
|
return ts >= this.builtTime;
|
|
|
|
}
|
2016-12-14 19:03:24 +08:00
|
|
|
|
2017-02-18 17:12:09 +08:00
|
|
|
build(options, compilation, resolver, fs, callback) {
|
|
|
|
this.built = true;
|
2017-05-04 10:19:54 +08:00
|
|
|
this.builtTime = Date.now();
|
2017-10-14 05:31:15 +08:00
|
|
|
this.resolveDependencies(fs, this.options, (err, dependencies) => {
|
2017-02-18 17:12:09 +08:00
|
|
|
if(err) return callback(err);
|
2013-02-01 01:00:22 +08:00
|
|
|
|
2017-05-05 00:37:25 +08:00
|
|
|
// Reset children
|
2017-11-06 20:02:35 +08:00
|
|
|
this.dependencies.length = 0;
|
|
|
|
this.blocks.length = 0;
|
2017-05-05 00:37:25 +08:00
|
|
|
|
|
|
|
// abort if something failed
|
|
|
|
// this will create an empty context
|
2017-02-18 18:09:18 +08:00
|
|
|
if(!dependencies) {
|
|
|
|
callback();
|
|
|
|
return;
|
2016-06-23 00:24:10 +08:00
|
|
|
}
|
2017-02-18 18:09:18 +08:00
|
|
|
|
2017-05-05 00:37:25 +08:00
|
|
|
// enhance dependencies with meta info
|
2017-02-18 18:09:18 +08:00
|
|
|
dependencies.forEach(dep => {
|
|
|
|
dep.loc = dep.userRequest;
|
2017-10-14 05:31:15 +08:00
|
|
|
dep.request = this.options.addon + dep.request;
|
2017-02-18 18:09:18 +08:00
|
|
|
});
|
|
|
|
|
2017-10-16 20:49:51 +08:00
|
|
|
if(this.options.mode === "sync" || this.options.mode === "eager") {
|
2017-05-05 00:37:25 +08:00
|
|
|
|
|
|
|
// if we have an sync or eager context
|
|
|
|
// just add all dependencies and continue
|
2017-02-18 17:12:09 +08:00
|
|
|
this.dependencies = dependencies;
|
2017-02-18 18:09:18 +08:00
|
|
|
|
2017-10-16 20:49:51 +08:00
|
|
|
} else if(this.options.mode === "lazy-once") {
|
2017-05-05 00:37:25 +08:00
|
|
|
|
|
|
|
// for the lazy-once mode create a new async dependency block
|
|
|
|
// and add that block to this context
|
|
|
|
if(dependencies.length > 0) {
|
2017-10-14 05:31:15 +08:00
|
|
|
const block = new AsyncDependenciesBlock(this.options.chunkName, this);
|
2017-05-05 00:37:25 +08:00
|
|
|
dependencies.forEach(dep => {
|
|
|
|
block.addDependency(dep);
|
|
|
|
});
|
|
|
|
this.addBlock(block);
|
|
|
|
}
|
|
|
|
|
2017-10-16 20:49:51 +08:00
|
|
|
} else if(this.options.mode === "weak" || this.options.mode === "async-weak") {
|
2017-07-07 19:26:48 +08:00
|
|
|
|
2017-07-26 20:49:37 +08:00
|
|
|
// we mark all dependencies as weak
|
|
|
|
dependencies.forEach(dep => dep.weak = true);
|
|
|
|
this.dependencies = dependencies;
|
2017-07-07 19:26:48 +08:00
|
|
|
|
2017-10-16 20:49:51 +08:00
|
|
|
} else if(this.options.mode === "lazy") {
|
2017-07-26 20:49:37 +08:00
|
|
|
// if we are lazy create a new async dependency block per dependency
|
2017-05-05 00:37:25 +08:00
|
|
|
// and add all blocks to this context
|
|
|
|
dependencies.forEach((dep, idx) => {
|
2017-10-14 05:31:15 +08:00
|
|
|
let chunkName = this.options.chunkName;
|
2017-05-05 00:37:25 +08:00
|
|
|
if(chunkName) {
|
|
|
|
if(!/\[(index|request)\]/.test(chunkName))
|
|
|
|
chunkName += "[index]";
|
|
|
|
chunkName = chunkName.replace(/\[index\]/g, idx);
|
|
|
|
chunkName = chunkName.replace(/\[request\]/g, Template.toPath(dep.userRequest));
|
|
|
|
}
|
|
|
|
const block = new AsyncDependenciesBlock(chunkName, dep.module, dep.loc);
|
|
|
|
block.addDependency(dep);
|
|
|
|
this.addBlock(block);
|
|
|
|
});
|
2017-10-16 20:49:51 +08:00
|
|
|
} else {
|
|
|
|
callback(new Error(`Unsupported mode "${this.options.mode}" in context`));
|
|
|
|
return;
|
2017-05-05 00:37:25 +08:00
|
|
|
}
|
2017-02-18 17:12:09 +08:00
|
|
|
callback();
|
2015-07-01 06:19:52 +08:00
|
|
|
});
|
2014-07-26 20:48:42 +08:00
|
|
|
}
|
2017-02-18 17:12:09 +08:00
|
|
|
|
2017-05-05 00:37:25 +08:00
|
|
|
getUserRequestMap(dependencies) {
|
2017-02-18 19:41:39 +08:00
|
|
|
// if we filter first we get a new array
|
|
|
|
// therefor we dont need to create a clone of dependencies explicitly
|
|
|
|
// therefore the order of this is !important!
|
2017-05-05 00:37:25 +08:00
|
|
|
return dependencies
|
2017-02-18 19:41:39 +08:00
|
|
|
.filter(dependency => dependency.module)
|
|
|
|
.sort((a, b) => {
|
|
|
|
if(a.userRequest === b.userRequest) {
|
|
|
|
return 0;
|
|
|
|
}
|
2017-02-18 17:12:09 +08:00
|
|
|
return a.userRequest < b.userRequest ? -1 : 1;
|
2017-11-08 18:32:05 +08:00
|
|
|
}).reduce((map, dep) => {
|
2017-02-18 19:41:39 +08:00
|
|
|
map[dep.userRequest] = dep.module.id;
|
|
|
|
return map;
|
2017-02-23 18:52:11 +08:00
|
|
|
}, Object.create(null));
|
2017-05-05 00:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
getSyncSource(dependencies, id) {
|
|
|
|
const map = this.getUserRequestMap(dependencies);
|
2017-02-18 19:41:39 +08:00
|
|
|
return `var map = ${JSON.stringify(map, null, "\t")};
|
|
|
|
function webpackContext(req) {
|
|
|
|
return __webpack_require__(webpackContextResolve(req));
|
|
|
|
};
|
|
|
|
function webpackContextResolve(req) {
|
|
|
|
var id = map[req];
|
2017-02-23 18:52:11 +08:00
|
|
|
if(!(id + 1)) // check for number or string
|
2017-02-18 19:41:39 +08:00
|
|
|
throw new Error("Cannot find module '" + req + "'.");
|
|
|
|
return id;
|
|
|
|
};
|
|
|
|
webpackContext.keys = function webpackContextKeys() {
|
|
|
|
return Object.keys(map);
|
|
|
|
};
|
|
|
|
webpackContext.resolve = webpackContextResolve;
|
|
|
|
module.exports = webpackContext;
|
|
|
|
webpackContext.id = ${JSON.stringify(id)};`;
|
|
|
|
}
|
|
|
|
|
2017-07-26 01:21:02 +08:00
|
|
|
getWeakSyncSource(dependencies, id) {
|
|
|
|
const map = this.getUserRequestMap(dependencies);
|
|
|
|
return `var map = ${JSON.stringify(map, null, "\t")};
|
|
|
|
function webpackContext(req) {
|
2017-07-26 20:49:37 +08:00
|
|
|
var id = webpackContextResolve(req);
|
|
|
|
if(!__webpack_require__.m[id])
|
|
|
|
throw new Error("Module '" + req + "' ('" + id + "') is not available (weak dependency)");
|
|
|
|
return __webpack_require__(id);
|
2017-07-26 01:21:02 +08:00
|
|
|
};
|
|
|
|
function webpackContextResolve(req) {
|
|
|
|
var id = map[req];
|
|
|
|
if(!(id + 1)) // check for number or string
|
|
|
|
throw new Error("Cannot find module '" + req + "'.");
|
|
|
|
return id;
|
|
|
|
};
|
|
|
|
webpackContext.keys = function webpackContextKeys() {
|
|
|
|
return Object.keys(map);
|
|
|
|
};
|
|
|
|
webpackContext.resolve = webpackContextResolve;
|
2017-07-26 21:13:39 +08:00
|
|
|
webpackContext.id = ${JSON.stringify(id)};
|
|
|
|
module.exports = webpackContext;`;
|
2017-07-26 01:21:02 +08:00
|
|
|
}
|
|
|
|
|
2017-07-26 20:49:37 +08:00
|
|
|
getAsyncWeakSource(dependencies, id) {
|
|
|
|
const map = this.getUserRequestMap(dependencies);
|
2017-07-26 01:21:02 +08:00
|
|
|
|
|
|
|
return `var map = ${JSON.stringify(map, null, "\t")};
|
|
|
|
function webpackAsyncContext(req) {
|
2017-07-26 20:49:37 +08:00
|
|
|
return webpackAsyncContextResolve(req).then(function(id) {
|
2017-07-26 01:21:02 +08:00
|
|
|
if(!__webpack_require__.m[id])
|
2017-07-26 20:49:37 +08:00
|
|
|
throw new Error("Module '" + req + "' ('" + id + "') is not available (weak dependency)");
|
|
|
|
return __webpack_require__(id);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
function webpackAsyncContextResolve(req) {
|
|
|
|
// Here Promise.resolve().then() is used instead of new Promise() to prevent
|
|
|
|
// uncatched exception popping up in devtools
|
|
|
|
return Promise.resolve().then(function() {
|
|
|
|
var id = map[req];
|
|
|
|
if(!(id + 1)) // check for number or string
|
|
|
|
throw new Error("Cannot find module '" + req + "'.");
|
|
|
|
return id;
|
2017-07-26 01:21:02 +08:00
|
|
|
});
|
|
|
|
};
|
|
|
|
webpackAsyncContext.keys = function webpackAsyncContextKeys() {
|
|
|
|
return Object.keys(map);
|
|
|
|
};
|
2017-07-26 20:49:37 +08:00
|
|
|
webpackAsyncContext.resolve = webpackAsyncContextResolve;
|
|
|
|
webpackAsyncContext.id = ${JSON.stringify(id)};
|
|
|
|
module.exports = webpackAsyncContext;`;
|
2017-07-26 01:21:02 +08:00
|
|
|
}
|
|
|
|
|
2017-05-05 00:37:25 +08:00
|
|
|
getEagerSource(dependencies, id) {
|
|
|
|
const map = this.getUserRequestMap(dependencies);
|
|
|
|
return `var map = ${JSON.stringify(map, null, "\t")};
|
|
|
|
function webpackAsyncContext(req) {
|
|
|
|
return webpackAsyncContextResolve(req).then(__webpack_require__);
|
|
|
|
};
|
|
|
|
function webpackAsyncContextResolve(req) {
|
2017-07-26 20:49:37 +08:00
|
|
|
// Here Promise.resolve().then() is used instead of new Promise() to prevent
|
|
|
|
// uncatched exception popping up in devtools
|
|
|
|
return Promise.resolve().then(function() {
|
2017-05-05 00:37:25 +08:00
|
|
|
var id = map[req];
|
|
|
|
if(!(id + 1)) // check for number or string
|
2017-07-26 20:49:37 +08:00
|
|
|
throw new Error("Cannot find module '" + req + "'.");
|
|
|
|
return id;
|
2017-05-05 00:37:25 +08:00
|
|
|
});
|
|
|
|
};
|
|
|
|
webpackAsyncContext.keys = function webpackAsyncContextKeys() {
|
|
|
|
return Object.keys(map);
|
|
|
|
};
|
|
|
|
webpackAsyncContext.resolve = webpackAsyncContextResolve;
|
2017-07-26 20:49:37 +08:00
|
|
|
webpackAsyncContext.id = ${JSON.stringify(id)};
|
|
|
|
module.exports = webpackAsyncContext;`;
|
2017-05-05 00:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
getLazyOnceSource(block, dependencies, id, outputOptions, requestShortener) {
|
|
|
|
const promise = DepBlockHelpers.getDepBlockPromise(block, outputOptions, requestShortener, "lazy-once context");
|
|
|
|
const map = this.getUserRequestMap(dependencies);
|
|
|
|
return `var map = ${JSON.stringify(map, null, "\t")};
|
|
|
|
function webpackAsyncContext(req) {
|
|
|
|
return webpackAsyncContextResolve(req).then(__webpack_require__);
|
|
|
|
};
|
|
|
|
function webpackAsyncContextResolve(req) {
|
|
|
|
return ${promise}.then(function() {
|
|
|
|
var id = map[req];
|
|
|
|
if(!(id + 1)) // check for number or string
|
|
|
|
throw new Error("Cannot find module '" + req + "'.");
|
|
|
|
return id;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
webpackAsyncContext.keys = function webpackAsyncContextKeys() {
|
|
|
|
return Object.keys(map);
|
|
|
|
};
|
|
|
|
webpackAsyncContext.resolve = webpackAsyncContextResolve;
|
2017-07-26 20:49:37 +08:00
|
|
|
webpackAsyncContext.id = ${JSON.stringify(id)};
|
|
|
|
module.exports = webpackAsyncContext;`;
|
2017-05-05 00:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
getLazySource(blocks, id) {
|
2017-02-18 20:01:29 +08:00
|
|
|
let hasMultipleOrNoChunks = false;
|
2017-02-18 19:41:39 +08:00
|
|
|
const map = blocks
|
|
|
|
.filter(block => block.dependencies[0].module)
|
2017-02-23 18:52:11 +08:00
|
|
|
.map((block) => ({
|
|
|
|
dependency: block.dependencies[0],
|
|
|
|
block: block,
|
|
|
|
userRequest: block.dependencies[0].userRequest
|
|
|
|
})).sort((a, b) => {
|
2017-02-18 17:12:09 +08:00
|
|
|
if(a.userRequest === b.userRequest) return 0;
|
|
|
|
return a.userRequest < b.userRequest ? -1 : 1;
|
2017-02-23 18:52:11 +08:00
|
|
|
}).reduce((map, item) => {
|
2017-02-18 19:41:39 +08:00
|
|
|
const chunks = item.block.chunks || [];
|
|
|
|
if(chunks.length !== 1) {
|
2017-02-18 20:01:29 +08:00
|
|
|
hasMultipleOrNoChunks = true;
|
2017-02-18 17:12:09 +08:00
|
|
|
}
|
2017-02-18 19:41:39 +08:00
|
|
|
map[item.userRequest] = [item.dependency.module.id]
|
|
|
|
.concat(chunks.map(chunk => chunk.id));
|
|
|
|
|
|
|
|
return map;
|
2017-02-23 18:52:11 +08:00
|
|
|
}, Object.create(null));
|
2017-02-18 19:41:39 +08:00
|
|
|
|
2017-02-18 20:01:29 +08:00
|
|
|
const requestPrefix = hasMultipleOrNoChunks ?
|
2017-02-18 19:41:39 +08:00
|
|
|
"Promise.all(ids.slice(1).map(__webpack_require__.e))" :
|
|
|
|
"__webpack_require__.e(ids[1])";
|
|
|
|
|
|
|
|
return `var map = ${JSON.stringify(map, null, "\t")};
|
|
|
|
function webpackAsyncContext(req) {
|
|
|
|
var ids = map[req];
|
|
|
|
if(!ids)
|
|
|
|
return Promise.reject(new Error("Cannot find module '" + req + "'."));
|
|
|
|
return ${requestPrefix}.then(function() {
|
|
|
|
return __webpack_require__(ids[0]);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
webpackAsyncContext.keys = function webpackAsyncContextKeys() {
|
|
|
|
return Object.keys(map);
|
|
|
|
};
|
2017-07-26 20:49:37 +08:00
|
|
|
webpackAsyncContext.id = ${JSON.stringify(id)};
|
|
|
|
module.exports = webpackAsyncContext;`;
|
2017-02-18 19:41:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
getSourceForEmptyContext(id) {
|
|
|
|
return `function webpackEmptyContext(req) {
|
|
|
|
throw new Error("Cannot find module '" + req + "'.");
|
|
|
|
}
|
|
|
|
webpackEmptyContext.keys = function() { return []; };
|
|
|
|
webpackEmptyContext.resolve = webpackEmptyContext;
|
|
|
|
module.exports = webpackEmptyContext;
|
|
|
|
webpackEmptyContext.id = ${JSON.stringify(id)};`;
|
|
|
|
}
|
|
|
|
|
2017-05-05 00:37:25 +08:00
|
|
|
getSourceForEmptyAsyncContext(id) {
|
|
|
|
return `function webpackEmptyAsyncContext(req) {
|
2017-07-26 20:49:37 +08:00
|
|
|
// Here Promise.resolve().then() is used instead of new Promise() to prevent
|
|
|
|
// uncatched exception popping up in devtools
|
|
|
|
return Promise.resolve().then(function() {
|
|
|
|
throw new Error("Cannot find module '" + req + "'.");
|
|
|
|
});
|
2017-05-05 00:37:25 +08:00
|
|
|
}
|
|
|
|
webpackEmptyAsyncContext.keys = function() { return []; };
|
|
|
|
webpackEmptyAsyncContext.resolve = webpackEmptyAsyncContext;
|
|
|
|
module.exports = webpackEmptyAsyncContext;
|
|
|
|
webpackEmptyAsyncContext.id = ${JSON.stringify(id)};`;
|
|
|
|
}
|
2017-02-18 19:41:39 +08:00
|
|
|
|
2017-05-05 00:37:25 +08:00
|
|
|
getSourceString(asyncMode, outputOptions, requestShortener) {
|
|
|
|
if(asyncMode === "lazy") {
|
|
|
|
if(this.blocks && this.blocks.length > 0) {
|
|
|
|
return this.getLazySource(this.blocks, this.id);
|
|
|
|
}
|
|
|
|
return this.getSourceForEmptyAsyncContext(this.id);
|
|
|
|
}
|
|
|
|
if(asyncMode === "eager") {
|
|
|
|
if(this.dependencies && this.dependencies.length > 0) {
|
|
|
|
return this.getEagerSource(this.dependencies, this.id);
|
|
|
|
}
|
|
|
|
return this.getSourceForEmptyAsyncContext(this.id);
|
2017-07-26 01:21:02 +08:00
|
|
|
}
|
|
|
|
if(asyncMode === "lazy-once") {
|
2017-05-05 00:37:25 +08:00
|
|
|
const block = this.blocks[0];
|
|
|
|
if(block) {
|
|
|
|
return this.getLazyOnceSource(block, block.dependencies, this.id, outputOptions, requestShortener);
|
|
|
|
}
|
|
|
|
return this.getSourceForEmptyAsyncContext(this.id);
|
|
|
|
}
|
2017-07-26 20:49:37 +08:00
|
|
|
if(asyncMode === "async-weak") {
|
|
|
|
if(this.dependencies && this.dependencies.length > 0) {
|
|
|
|
return this.getAsyncWeakSource(this.dependencies, this.id);
|
2017-07-26 01:21:02 +08:00
|
|
|
}
|
|
|
|
return this.getSourceForEmptyAsyncContext(this.id);
|
|
|
|
}
|
|
|
|
if(asyncMode === "weak") {
|
|
|
|
if(this.dependencies && this.dependencies.length > 0) {
|
|
|
|
return this.getWeakSyncSource(this.dependencies, this.id);
|
|
|
|
}
|
|
|
|
}
|
2017-05-05 00:37:25 +08:00
|
|
|
if(this.dependencies && this.dependencies.length > 0) {
|
|
|
|
return this.getSyncSource(this.dependencies, this.id);
|
2017-02-18 17:12:09 +08:00
|
|
|
}
|
2017-02-18 19:41:39 +08:00
|
|
|
return this.getSourceForEmptyContext(this.id);
|
|
|
|
}
|
|
|
|
|
|
|
|
getSource(sourceString) {
|
2017-02-18 17:12:09 +08:00
|
|
|
if(this.useSourceMap) {
|
2017-02-18 19:41:39 +08:00
|
|
|
return new OriginalSource(sourceString, this.identifier());
|
2017-02-18 17:12:09 +08:00
|
|
|
}
|
2017-02-18 19:41:39 +08:00
|
|
|
return new RawSource(sourceString);
|
|
|
|
}
|
|
|
|
|
2017-05-05 00:37:25 +08:00
|
|
|
source(dependencyTemplates, outputOptions, requestShortener) {
|
2017-02-18 19:41:39 +08:00
|
|
|
return this.getSource(
|
2017-10-16 20:49:51 +08:00
|
|
|
this.getSourceString(this.options.mode, outputOptions, requestShortener)
|
2017-02-18 19:41:39 +08:00
|
|
|
);
|
2013-10-14 19:59:44 +08:00
|
|
|
}
|
2013-01-31 01:49:25 +08:00
|
|
|
|
2017-02-18 17:12:09 +08:00
|
|
|
size() {
|
2017-02-18 18:13:39 +08:00
|
|
|
// base penalty
|
2017-02-28 19:07:27 +08:00
|
|
|
const initialSize = 160;
|
2017-02-18 18:13:39 +08:00
|
|
|
|
|
|
|
// if we dont have dependencies we stop here.
|
2017-02-19 08:00:06 +08:00
|
|
|
return this.dependencies
|
2017-02-28 19:07:27 +08:00
|
|
|
.reduce((size, dependency) => size + 5 + dependency.userRequest.length, initialSize);
|
2017-02-18 17:12:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-16 18:16:07 +08:00
|
|
|
Object.defineProperty(ContextModule.prototype, "recursive", {
|
|
|
|
configurable: false,
|
|
|
|
get: util.deprecate(function() {
|
|
|
|
return this.options.recursive;
|
|
|
|
}, "ContextModule.recursive has been moved to ContextModule.options.recursive"),
|
|
|
|
set: util.deprecate(function(value) {
|
|
|
|
this.options.recursive = value;
|
|
|
|
}, "ContextModule.recursive has been moved to ContextModule.options.recursive")
|
|
|
|
});
|
|
|
|
|
|
|
|
Object.defineProperty(ContextModule.prototype, "regExp", {
|
|
|
|
configurable: false,
|
|
|
|
get: util.deprecate(function() {
|
|
|
|
return this.options.regExp;
|
|
|
|
}, "ContextModule.regExp has been moved to ContextModule.options.regExp"),
|
|
|
|
set: util.deprecate(function(value) {
|
|
|
|
this.options.regExp = value;
|
|
|
|
}, "ContextModule.regExp has been moved to ContextModule.options.regExp")
|
|
|
|
});
|
|
|
|
|
|
|
|
Object.defineProperty(ContextModule.prototype, "addon", {
|
|
|
|
configurable: false,
|
|
|
|
get: util.deprecate(function() {
|
|
|
|
return this.options.addon;
|
|
|
|
}, "ContextModule.addon has been moved to ContextModule.options.addon"),
|
|
|
|
set: util.deprecate(function(value) {
|
|
|
|
this.options.addon = value;
|
|
|
|
}, "ContextModule.addon has been moved to ContextModule.options.addon")
|
|
|
|
});
|
|
|
|
|
|
|
|
Object.defineProperty(ContextModule.prototype, "async", {
|
|
|
|
configurable: false,
|
|
|
|
get: util.deprecate(function() {
|
2017-10-16 20:49:51 +08:00
|
|
|
return this.options.mode;
|
|
|
|
}, "ContextModule.async has been moved to ContextModule.options.mode"),
|
2017-10-16 18:16:07 +08:00
|
|
|
set: util.deprecate(function(value) {
|
2017-10-16 20:49:51 +08:00
|
|
|
this.options.mode = value;
|
|
|
|
}, "ContextModule.async has been moved to ContextModule.options.mode")
|
2017-10-16 18:16:07 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
Object.defineProperty(ContextModule.prototype, "chunkName", {
|
|
|
|
configurable: false,
|
|
|
|
get: util.deprecate(function() {
|
|
|
|
return this.options.chunkName;
|
|
|
|
}, "ContextModule.chunkName has been moved to ContextModule.options.chunkName"),
|
|
|
|
set: util.deprecate(function(value) {
|
|
|
|
this.options.chunkName = value;
|
|
|
|
}, "ContextModule.chunkName has been moved to ContextModule.options.chunkName")
|
|
|
|
});
|
|
|
|
|
2017-02-18 17:12:09 +08:00
|
|
|
module.exports = ContextModule;
|