-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathmain.js
114 lines (95 loc) · 3.34 KB
/
main.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
/**
* Pre-render (https://github.com/kriasoft/pre-render)
*
* Copyright © 2017-present Kriasoft. All rights reserved.
*
* This source code is licensed under the MIT license found in the
* LICENSE.txt file in the root directory of this source tree.
*/
/* eslint-disable no-restricted-syntax, no-await-in-loop */
const fs = require('fs');
const path = require('path');
const makeDir = require('make-dir');
const mimeTypes = require('mime-types');
const ChromeLauncher = require('chrome-launcher');
const CDP = require('chrome-remote-interface');
const log = require('lighthouse-logger');
async function render(src, urls = ['/']) {
if (!src) {
throw new Error('Must provide a path string.');
}
if (!urls || !Array.isArray(urls) || !urls.length) {
throw new Error('Must provide a list of relative URLs.');
}
let chrome;
let client;
try {
log.setLevel('info');
// Launch a new instance of the headless Chrome browser
// https://developers.google.com/web/updates/2017/04/headless-chrome
chrome = await ChromeLauncher.launch({
chromeFlags: [
'--window-size=1366,768',
'--disable-gpu',
// '--disable-web-security',
// '--user-data-dir',
'--headless',
],
});
// Configure Chrome DevTools Protocol
// https://github.com/cyrus-and/chrome-remote-interface
// https://github.com/cyrus-and/chrome-remote-interface/wiki
client = await CDP({ port: chrome.port });
const { Page, Network, Runtime, Security } = client;
await Network.setRequestInterceptionEnabled({ enabled: true });
// await Security.setOverrideCertificateErrors({ override: true });
await Promise.all([
Page.enable(),
Network.enable(),
Security.enable(),
Runtime.enable(),
]);
Network.requestIntercepted(({ interceptionId, request }) => {
// console.log(`${interceptionId} ${request.method} ${request.url}`);
if (request.url.startsWith('http://localhost:3000/')) {
const pathname = request.url.substr(21);
const file =
pathname === '/' ? `${src}/index.html` : `${src}${pathname}`;
const response = [
'HTTP/1.1 200 OK',
`Content-Type: ${mimeTypes.contentType(path.extname(file))}`,
'\r\n',
fs.readFileSync(file, 'utf8'),
].join('\r\n');
Network.continueInterceptedRequest({
interceptionId,
rawResponse: new Buffer(response).toString('base64'),
});
} else {
Network.continueInterceptedRequest({ interceptionId });
}
});
// Security.certificateError(({ eventId }) => {
// Security.handleCertificateError({ eventId, action: 'continue' });
// });
await Page.navigate({ url: 'http://localhost:3000/' });
await Page.loadEventFired();
// eslint-disable-next-line no-restricted-syntax, no-await-in-loop
for (const url of urls) {
const filename =
url === '/' ? `${src}/index.html` : `${src}/${url}/index.html`;
if (url !== '/') {
await makeDir(path.dirname(filename));
}
const { result } = await Runtime.evaluate({
expression: `window.prerender("${url}")`,
awaitPromise: true,
});
fs.writeFileSync(filename, result.value, 'utf8');
}
} finally {
if (client) client.close();
if (chrome) chrome.kill();
}
}
module.exports = render;