webpack/lib/optimize/OccurrenceOrderPlugin.js

115 lines
3.5 KiB
JavaScript
Raw Normal View History

/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
"use strict";
2015-07-13 06:20:09 +08:00
class OccurrenceOrderPlugin {
constructor(preferEntry) {
if(preferEntry !== undefined && typeof preferEntry !== "boolean") {
throw new Error("Argument should be a boolean.\nFor more info on this plugin, see https://webpack.js.org/plugins/");
}
this.preferEntry = preferEntry;
}
apply(compiler) {
const preferEntry = this.preferEntry;
2017-12-06 22:01:25 +08:00
compiler.hooks.compilation.tap("OccurrenceOrderPlugin", (compilation) => {
compilation.hooks.optimizeModuleOrder.tap("OccurrenceOrderPlugin", (modules) => {
2017-06-01 21:42:57 +08:00
const occursInInitialChunksMap = new Map();
const occursInAllChunksMap = new Map();
const initialChunkChunkMap = new Map();
const entryCountMap = new Map();
modules.forEach(m => {
let initial = 0;
let entry = 0;
2017-04-19 04:30:18 +08:00
m.forEachChunk(c => {
2017-06-01 21:42:57 +08:00
if(c.isInitial()) initial++;
if(c.entryModule === m) entry++;
2017-04-19 04:30:18 +08:00
});
2017-06-01 21:42:57 +08:00
initialChunkChunkMap.set(m, initial);
entryCountMap.set(m, entry);
});
2015-07-13 06:20:09 +08:00
2017-06-01 21:42:57 +08:00
const countOccursInEntry = (sum, r) => {
if(!r.module) return sum;
return sum + initialChunkChunkMap.get(r.module);
};
const countOccurs = (sum, r) => {
if(!r.module) return sum;
let factor = 1;
if(typeof r.dependency.getNumberOfIdOccurrences === "function")
factor = r.dependency.getNumberOfIdOccurrences();
if(factor === 0) return sum;
return sum + factor * r.module.getNumberOfChunks();
2017-06-01 21:42:57 +08:00
};
2015-07-13 06:20:09 +08:00
2017-06-01 21:42:57 +08:00
if(preferEntry) {
modules.forEach(m => {
const result = m.reasons.reduce(countOccursInEntry, 0) + initialChunkChunkMap.get(m) + entryCountMap.get(m);
occursInInitialChunksMap.set(m, result);
2017-04-19 04:30:18 +08:00
});
}
2017-06-01 21:42:57 +08:00
const originalOrder = new Map();
let i = 0;
2017-06-01 21:42:57 +08:00
modules.forEach(m => {
const result = m.reasons.reduce(countOccurs, 0) + m.getNumberOfChunks() + entryCountMap.get(m);
occursInAllChunksMap.set(m, result);
originalOrder.set(m, i++);
2017-06-01 21:42:57 +08:00
});
modules.sort((a, b) => {
if(preferEntry) {
2017-06-01 21:42:57 +08:00
const aEntryOccurs = occursInInitialChunksMap.get(a);
const bEntryOccurs = occursInInitialChunksMap.get(b);
if(aEntryOccurs > bEntryOccurs) return -1;
if(aEntryOccurs < bEntryOccurs) return 1;
}
2017-06-01 21:42:57 +08:00
const aOccurs = occursInAllChunksMap.get(a);
const bOccurs = occursInAllChunksMap.get(b);
if(aOccurs > bOccurs) return -1;
if(aOccurs < bOccurs) return 1;
const orgA = originalOrder.get(a);
const orgB = originalOrder.get(b);
return orgB - orgA;
});
});
2017-12-06 22:01:25 +08:00
compilation.hooks.optimizeChunkOrder.tap("OccurrenceOrderPlugin", (chunks) => {
2017-06-01 21:42:57 +08:00
const occursInInitialChunksMap = new Map();
const originalOrder = new Map();
2017-06-01 21:42:57 +08:00
let i = 0;
2017-06-01 21:42:57 +08:00
chunks.forEach(c => {
const result = c.getParents().reduce((sum, p) => {
2017-06-01 21:42:57 +08:00
if(p.isInitial()) return sum + 1;
return sum;
}, 0);
occursInInitialChunksMap.set(c, result);
originalOrder.set(c, i++);
2017-06-01 21:42:57 +08:00
});
2017-11-08 18:32:05 +08:00
const occurs = c => {
return c.getNumberOfBlocks();
2017-11-08 18:32:05 +08:00
};
2017-06-01 21:42:57 +08:00
chunks.sort((a, b) => {
2017-06-01 21:42:57 +08:00
const aEntryOccurs = occursInInitialChunksMap.get(a);
const bEntryOccurs = occursInInitialChunksMap.get(b);
if(aEntryOccurs > bEntryOccurs) return -1;
if(aEntryOccurs < bEntryOccurs) return 1;
const aOccurs = occurs(a);
const bOccurs = occurs(b);
if(aOccurs > bOccurs) return -1;
if(aOccurs < bOccurs) return 1;
const orgA = originalOrder.get(a);
const orgB = originalOrder.get(b);
return orgB - orgA;
});
});
});
}
}
module.exports = OccurrenceOrderPlugin;