forked from GoogleChrome/lighthouse
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathjs-bundles.js
122 lines (101 loc) · 4.23 KB
/
js-bundles.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
/**
* @license Copyright 2020 The Lighthouse Authors. All Rights Reserved.
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
*/
'use strict';
const log = require('lighthouse-logger');
const makeComputedArtifact = require('./computed-artifact.js');
const SDK = require('../lib/cdt/SDK.js');
/**
* Calculate the number of bytes contributed by each source file
* @param {LH.Artifacts.Bundle['map']} map
* @param {string} content
* @return {LH.Artifacts.Bundle['sizes']}
*/
function computeGeneratedFileSizes(map, content) {
const lines = content.split('\n');
/** @type {Record<string, number>} */
const files = {};
const totalBytes = content.length;
let unmappedBytes = totalBytes;
// If the map + contents don't line up, return a result that
// denotes nothing is mapped.
const failureResult = {files: {}, unmappedBytes, totalBytes};
// @ts-ignore: This function is added in SDK.js. This will eventually be added to CDT.
map.computeLastGeneratedColumns();
for (const mapping of map.mappings()) {
const source = mapping.sourceURL;
const lineNum = mapping.lineNumber;
const colNum = mapping.columnNumber;
// @ts-ignore: `lastColumnNumber` is not on types yet. This will eventually be added to CDT.
const lastColNum = /** @type {number=} */ (mapping.lastColumnNumber);
// Webpack sometimes emits null mappings.
// https://github.com/mozilla/source-map/pull/303
if (!source) continue;
// Lines and columns are zero-based indices. Visually, lines are shown as a 1-based index.
const line = lines[lineNum];
if (line === null) {
log.error('JSBundles', `${map.url()} mapping for line out of bounds: ${lineNum + 1}`);
return failureResult;
}
if (colNum > line.length) {
// eslint-disable-next-line max-len
log.error('JSBundles', `${map.url()} mapping for column out of bounds: ${lineNum + 1}:${colNum}`);
return failureResult;
}
let mappingLength = 0;
if (lastColNum !== undefined) {
if (lastColNum > line.length) {
// eslint-disable-next-line max-len
log.error('JSBundles', `${map.url()} mapping for last column out of bounds: ${lineNum + 1}:${lastColNum}`);
return failureResult;
}
mappingLength = lastColNum - colNum;
} else {
// Add +1 to account for the newline.
mappingLength = line.length - colNum + 1;
}
files[source] = (files[source] || 0) + mappingLength;
unmappedBytes -= mappingLength;
}
return {
files,
unmappedBytes,
totalBytes,
};
}
class JSBundles {
/**
* @param {Pick<LH.Artifacts, 'SourceMaps'|'ScriptElements'>} artifacts
*/
static async compute_(artifacts) {
const {SourceMaps, ScriptElements} = artifacts;
/** @type {LH.Artifacts.Bundle[]} */
const bundles = [];
// Collate map and script, compute file sizes.
for (const SourceMap of SourceMaps) {
if (!SourceMap.map) continue;
const {scriptUrl, map: rawMap} = SourceMap;
if (!rawMap.mappings) continue;
const scriptElement = ScriptElements.find(s => s.src === scriptUrl);
if (!scriptElement) continue;
const compiledUrl = SourceMap.scriptUrl || 'compiled.js';
const mapUrl = SourceMap.sourceMapUrl || 'compiled.js.map';
// @ts-ignore: CDT expects undefined properties to be explicit.
const rawMapForCdt = /** @type {any} */ (rawMap);
const map = new SDK.TextSourceMap(compiledUrl, mapUrl, rawMapForCdt);
const content = scriptElement && scriptElement.content ? scriptElement.content : '';
const sizes = computeGeneratedFileSizes(map, content);
const bundle = {
rawMap,
script: scriptElement,
map,
sizes,
};
bundles.push(bundle);
}
return bundles;
}
}
module.exports = makeComputedArtifact(JSBundles);