forked from rrweb-io/rrweb
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrepl.ts
135 lines (116 loc) · 4.2 KB
/
repl.ts
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
123
124
125
126
127
128
129
130
131
132
133
134
135
import * as fs from 'fs';
import * as path from 'path';
import * as puppeteer from 'puppeteer';
import { eventWithTime } from '../src/types';
var http = require("http");
var WebSocketServer = require("websocket").server;
var server = http.createServer(function (request, response) {
});
server.listen(1337, function () {
console.log("listening at ", new Date());
});
const wsServer = new WebSocketServer({
httpServer: server
//autoAcceptConnections: true
});
wsServer.on("request", function (request) {
var connection = request.accept(null, request.origin);
console.log("request at ", new Date());
connection.on("message", function (message) {
if (message.type === "utf8") {
console.log(message.utf8Data);
}
});
});
process
.on('uncaughtException', error => {
console.error(error);
})
.on('unhandledRejection', error => {
console.error(error);
});
function getCode(): string {
const bundlePath = path.resolve(__dirname, '../dist/rrweb.min.js');
return fs.readFileSync(bundlePath, 'utf8');
}
function saveEvents(events: eventWithTime[]) {
const tempFolder = path.join(__dirname, '../temp');
if (!fs.existsSync(tempFolder)) {
fs.mkdirSync(tempFolder);
}
const time = new Date()
.toISOString()
.replace(/[-|:]/g, '_')
.replace(/\..+/, '');
const content = `
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<meta http-equiv="X-UA-Compatible" content="ie=edge" />
<title>Record @${time}</title>
<link rel="stylesheet" href="../dist/rrweb.min.css" />
</head>
<body>
<script src="../dist/rrweb.min.js"></script>
<script>
/*<!--*/
const events = ${JSON.stringify(events).replace(
/<\/script>/g,
'<\\/script>',
)};
/*-->*/
const replayer = new rrweb.Replayer(events);
replayer.play();
</script>
</body>
</html>
`;
fs.writeFileSync(path.resolve(tempFolder, `replay_${time}.html`), content);
fs.writeFileSync(path.resolve(tempFolder, `events_${time}.json`), JSON.stringify(events));
console.log(`Saved at ${tempFolder} -> ${time}`);
}
var writeStream = fs.createWriteStream(path.join(__dirname, '../rec'), {
flags: 'a'
});
function write(event: eventWithTime) {
writeStream.write(JSON.stringify(event) + "\n");
}
(async () => {
let events: eventWithTime[] = [];
const code = getCode();
const url = process.argv[2].match(/\w+\:\/\//) ? process.argv[2] : 'http://' + process.argv[2];
const injectCode = `;${code}
window.__IS_RECORDING__ = true
rrweb.record({
emit: event => window._replLog(event)
});
`;
const browser = await puppeteer.launch({
headless: false,
defaultViewport: null,
args: ['--start-maximized'],
});
const pages = await browser.pages();
const page = pages[0];
await page.goto(url/*, {
waitUntil: 'domcontentloaded',
}*/);
await page.exposeFunction('_replLog', (event: eventWithTime) => {
events.push(event);
write(event);
});
await page.evaluate(injectCode);
page.on('framenavigated', async () => {
const isRecording = await page.evaluate('window.__IS_RECORDING__');
if (!isRecording) {
await page.evaluate(injectCode);
}
});
browser.once('disconnected', async () => {
saveEvents(events);
writeStream.close();
//process.exit();
});
})();