1 | const 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 |
|
8 | const 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 |
|
33 | const sqr = (x) => x * x;
|
34 | const mean = (xs) => xs.reduce((acc, x) => acc + x, 0) / xs.length;
|
35 | const median = (xs) => xs.sort()[Math.floor(xs.length / 2)];
|
36 | const variance = (xs, mean) =>
|
37 | xs.reduce((acc, x) => acc + sqr(x - mean), 0) / (xs.length - 1);
|
38 | const 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 |
|
47 | module.exports.getModuleName = (module) => module.userRequest;
|
48 |
|
49 | module.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 |
|
64 | module.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 |
|
77 | module.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 |
|
91 | module.exports.getTotalActiveTime = (group) => {
|
92 | const mergedRanges = mergeRanges(group);
|
93 | return mergedRanges.reduce((acc, range) => acc + range.end - range.start, 0);
|
94 | };
|
95 |
|
96 | const 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 | };
|
132 | module.exports.prependLoader = prependLoader;
|
133 |
|
134 | module.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 |
|
154 | const toCamelCase = (s) => s.replace(/(\-\w)/g, (m) => m[1].toUpperCase());
|
155 | module.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 | };
|