UNPKG

4.6 kBJavaScriptView Raw
1const isEqual = (x, y) =>
2 Array.isArray(x)
3 ? Array.isArray(y) &&
4 x.every((xi) => y.includes(xi)) &&
5 y.every((yi) => x.includes(yi))
6 : x === y;
7
8const mergeRanges = (rangeList) => {
9 const mergedQueue = [];
10 const inputQueue = [...rangeList];
11 while (inputQueue.length) {
12 const cur = inputQueue.pop();
13 const overlapIndex = mergedQueue.findIndex(
14 (item) =>
15 (item.start >= cur.start && item.start <= cur.end) ||
16 (item.end >= cur.start && item.end <= cur.end)
17 );
18
19 if (overlapIndex === -1) {
20 mergedQueue.push(cur);
21 } else {
22 const toMerge = mergedQueue.splice(overlapIndex, 1)[0];
23 inputQueue.push({
24 start: Math.min(cur.start, cur.end, toMerge.start, toMerge.end),
25 end: Math.max(cur.start, cur.end, toMerge.start, toMerge.end),
26 });
27 }
28 }
29
30 return mergedQueue;
31};
32
33const sqr = (x) => x * x;
34const mean = (xs) => xs.reduce((acc, x) => acc + x, 0) / xs.length;
35const median = (xs) => xs.sort()[Math.floor(xs.length / 2)];
36const variance = (xs, mean) =>
37 xs.reduce((acc, x) => acc + sqr(x - mean), 0) / (xs.length - 1);
38const range = (xs) =>
39 xs.reduce(
40 (acc, x) => ({
41 start: Math.min(x, acc.start),
42 end: Math.max(x, acc.end),
43 }),
44 { start: Number.POSITIVE_INFINITY, end: Number.NEGATIVE_INFINITY }
45 );
46
47module.exports.getModuleName = (module) => module.userRequest;
48
49module.exports.getLoaderNames = (loaders) =>
50 loaders && loaders.length
51 ? loaders
52 .map((l) => l.loader || l)
53 .map((l) =>
54 l
55 .replace(/\\/g, "/")
56 .replace(
57 /^.*\/node_modules\/(@[a-z0-9][\w-.]+\/[a-z0-9][\w-.]*|[^\/]+).*$/,
58 (_, m) => m
59 )
60 )
61 .filter((l) => !l.includes("speed-measure-webpack-plugin"))
62 : ["modules with no loaders"];
63
64module.exports.groupBy = (key, arr) => {
65 const groups = [];
66 (arr || []).forEach((arrItem) => {
67 const groupItem = groups.find((poss) =>
68 isEqual(poss[0][key], arrItem[key])
69 );
70 if (groupItem) groupItem.push(arrItem);
71 else groups.push([arrItem]);
72 });
73
74 return groups;
75};
76
77module.exports.getAverages = (group) => {
78 const durationList = group.map((cur) => cur.end - cur.start);
79
80 const averages = {};
81 averages.dataPoints = group.length;
82 averages.median = median(durationList);
83 averages.mean = Math.round(mean(durationList));
84 averages.range = range(durationList);
85 if (group.length > 1)
86 averages.variance = Math.round(variance(durationList, averages.mean));
87
88 return averages;
89};
90
91module.exports.getTotalActiveTime = (group) => {
92 const mergedRanges = mergeRanges(group);
93 return mergedRanges.reduce((acc, range) => acc + range.end - range.start, 0);
94};
95
96const prependLoader = (rules) => {
97 if (!rules) return rules;
98 if (Array.isArray(rules)) return rules.map(prependLoader);
99
100 if (rules.loader) {
101 rules.use = [rules.loader];
102 if (rules.options) {
103 rules.use[0] = { loader: rules.loader, options: rules.options };
104 delete rules.options;
105 }
106 delete rules.loader;
107 }
108
109 if (rules.use) {
110 if (!Array.isArray(rules.use)) rules.use = [rules.use];
111 rules.use.unshift("speed-measure-webpack-plugin/loader");
112 }
113
114 if (rules.oneOf) {
115 rules.oneOf = prependLoader(rules.oneOf);
116 }
117 if (rules.rules) {
118 rules.rules = prependLoader(rules.rules);
119 }
120 if (Array.isArray(rules.resource)) {
121 rules.resource = prependLoader(rules.resource);
122 }
123 if (rules.resource && rules.resource.and) {
124 rules.resource.and = prependLoader(rules.resource.and);
125 }
126 if (rules.resource && rules.resource.or) {
127 rules.resource.or = prependLoader(rules.resource.or);
128 }
129
130 return rules;
131};
132module.exports.prependLoader = prependLoader;
133
134module.exports.hackWrapLoaders = (loaderPaths, callback) => {
135 const wrapReq = (reqMethod) => {
136 return function () {
137 const ret = reqMethod.apply(this, arguments);
138 if (loaderPaths.includes(arguments[0])) {
139 if (ret.__smpHacked) return ret;
140 ret.__smpHacked = true;
141 return callback(ret, arguments[0]);
142 }
143 return ret;
144 };
145 };
146
147 if (typeof System === "object" && typeof System.import === "function") {
148 System.import = wrapReq(System.import);
149 }
150 const Module = require("module");
151 Module.prototype.require = wrapReq(Module.prototype.require);
152};
153
154const toCamelCase = (s) => s.replace(/(\-\w)/g, (m) => m[1].toUpperCase());
155module.exports.tap = (obj, hookName, func) => {
156 if (obj.hooks) {
157 return obj.hooks[toCamelCase(hookName)].tap("smp", func);
158 }
159 return obj.plugin(hookName, func);
160};