背景
对于一些比较重要的项目,我们可能需要对这些项目中的某些网页进行特殊处理,比如记录的用户行为,这样做是有好处的,通过分析用户的行为从而对项目的流程进行优化;采集用户遇到的 bug 的操作路径,尤其是当我们无法复现生产环境的 BUG 时,录制用户行为对开发人员有很大的帮助,那么该如何实现录制用户行为的功能呢?
关于录制用户行为需要注意两点:
- 必须要做到用户无感知才行,如果让用户决定是否录制,那其实就无法记录用户行为了,一般来说用户是会拒绝录制的。
- 要支持回放,不然录制就没意义了,后续也无法分析用户的行为
录制方案
提到录制,我们都会想到两个解决方案:WebRTC 和 rrweb,如果还有其他更好的实现方案以及第三方库,可以推荐给我。
方案一:WebRTC
WebRTC(Web Real-Time Communications)是 Google 公司开源的一项实时通讯技术,它允许网络应用或者站点,在不借助中间媒介的情况下,建立浏览器之间点对点的连接,实现视频流、音频流或者其他任意数据的传输。
使用 navigator 的一个 API 来访问流媒体设备(如摄像机或麦克风),它允许网页访问摄像头或麦克风,并捕获实时流视频/音频。
1 function getUserMedia() {
2 navigator.mediaDevices.getUserMedia({
3 video: {facingMode: facingMode},
4 audio: true
5 }).then(stream => {
6 localStream = stream
7 const localVideo = document.getElementById('localVideo');
8 localVideo.srcObject = stream;
9 });
10 }
使用 WebRTC 一个很大的问题就是,它无法做到无感知录制,浏览器会询问用户是否同意,所以使用 WebRTC 肯定无法满足我们的需求,那就只能采用 rrweb 了。
方案二:rrweb
rrweb 是 record and replay the web 的简写,旨在利用现代浏览器所提供的强大 API 录制并回放任意 web 界面中的用户操作,rrweb 的官网:www.rrweb.io/ ,废话不多说,开始实践,通过一个例子来学习 rrweb 的使用!
页面结构
安装插件:
1pnpm i rrweb rrweb-player
2pnpm i @rrweb/types -D
记得引入 rrweb-player 的样式:import ‘rrweb-player/dist/style.css’;
核心逻辑
我这里是将录制的全部逻辑封装成 hooks 使用:use-record.ts
,完整代码放到最后
核心代码不多,十几行搞定,录制和回放时调用下第三方插件即可。
效果展示
随便写了个 Demo,来看下录制与回放的效果:
rrweb原理浅析
DOM快照
什么是 DOM 快照:⻚⾯中的视图状态可以通过 DOM 树的形式描述,所以当我们尝试录制⼀个⻚⾯时,我们可以记录 DOM 树在各个时间点上的状态。 记录每一时刻页面的DOM状态,回放的时候根据时间点显示即可。
其实 rrweb 录制的并不是视频,而是一系列的 DOM 结构,我们打印下 event
参数:
录制完成后可以得到 DOM 快照集合:eventList
,回放时 rrwebPlayer 再将 eventList
渲染出来。
优化DOM的记录
如果将每时每刻的 DOM 状态都记录下来,对于 DOM 数据庞大的情况,就有可能出现性能问题,rrweb 内部其实是对记录 DOM 的过程进行了优化。
- 记录初始页面的 DOM 状态,后续收集某时刻某个 DOM 的变化作为一个增量快照,在原先快照的基础上,不断加入根据行为解析的 DOM 数据,构建了后续的快照。
- 对于鼠标移动,页面滚动等事件进行了节流
- 压缩数据
组成部分
rrweb 主要包含下面三个部分:
- rrweb-snapshot,包含 snapshot 和 rebuild 两个功能。 snapshot 用于将 DOM 及其状态转化为可序列化的数据结构并添加唯一标识; rebuild 则是将 snapshot 记录的数据结构重建为对应的 DOM,并插入文档中
- rrweb,包含 record 和 replay 两个功能。 record 用于记录 DOM 中的所有变更; replay 则是将记录的变更按照对应的时间一一重放
- rrweb-player,为 rrweb 提供一套 UI 控件,提供基于图形用户界面的暂停、快进、拖拽至任意时间点播放等功能
后续优化
录制完成之后,需要将数据传给后台,便于在后台管理系统中查看,这就会涉及到传输数据量过大的情况,简单录制 7s,传给后台的数据量为 300 多 KB,如果是用户的操作时间过长,传输的数据量过多,会增加服务器的压力,所以这里需要进行优化。
压缩数据
rrweb 有提供压缩的功能,压缩后为 100KB 左右:
开启压缩:
后台展示回放时需要解码操作:
非全量录制
除了压缩数据量,还可以考虑只在重要的页面/核心功能中去录制用户的行为,而不是全量录制,尽量减少数据量,这样也有助于后台管理人员的查看,毕竟谁没事会去看这么长的回放。
上传时机优化
可以考虑每隔一段时间向服务器上传录制的数据,以减轻浏览器和服务器的压力,或者是在浏览器空闲的时候,进行数据的上传。
只上传报错
传输数据量过大会影响页面的性能,所以要尽可能减少数据量。
如果是只想录制报错的回放,便于复现排查生产环境的 BUG 的话,在上传 eventList
的前,判断下录制的内容里是否有报错,如果有报错再上传;如果是想分析用户行为,用于优化页面逻辑,那就只能都上传了。
关于 eventList
中是否有报错的记录,需要研究下能不能实现,我感觉是比较有难度的,毕竟报错的定义很广泛,这就要涉及到各种类型报错的处理。
当然录制用户行为会涉及到用户的隐私问题,所以还是得慎重考虑才行。
完整Demo代码
rrweb-demo.vue
1<template>
2 <main>
3 <header>
4 <el-button @click="onRecord" type="primary">录制</el-button>
5 <el-button @click="onReplay" type="success">回放</el-button>
6 <el-button @click="goBack">返回</el-button>
7 </header>
8
9 <section v-if="showReplay" ref="replayer"></section>
10
11 <section v-else>
12 <el-form
13 ref="ruleFormRef"
14 style="max-width: 600px"
15 :model="ruleForm"
16 status-icon
17 label-width="auto"
18 >
19 <el-form-item label="用户名">
20 <el-input v-model="ruleForm.name" autocomplete="off" />
21 </el-form-item>
22 <el-form-item label="密码">
23 <el-input
24 v-model="ruleForm.pass"
25 type="password"
26 autocomplete="off"
27 />
28 </el-form-item>
29 <el-form-item label="年龄">
30 <el-input v-model.number="ruleForm.age" />
31 </el-form-item>
32 <el-form-item style="margin-left: 55px">
33 <el-button type="primary"> 提交 </el-button>
34 <el-button>重置</el-button>
35 </el-form-item>
36 </el-form>
37 </section>
38 </main>
39</template>
40
41<script lang="ts" setup>
42import { reactive, ref } from 'vue';
43import type { FormInstance } from 'element-plus';
44import { useRecord } from '@/hooks/use-record';
45import 'rrweb-player/dist/style.css';
46
47const { replayer, showReplay, onRecord, onReplay, goBack } = useRecord();
48
49const ruleFormRef = ref<FormInstance>();
50const ruleForm = reactive({
51 pass: '',
52 name: '',
53 age: '',
54});
55</script>
56
57<style scoped lang="less">
58main {
59 display: flex;
60 flex-direction: column;
61 justify-content: center;
62 align-items: center;
63 width: 100%;
64 height: 100%;
65 margin-top: 50px;
66 header {
67 margin-bottom: 20px;
68 }
69 section {
70 border: 1px solid rgb(198, 194, 194);
71 padding: 20px;
72 border-radius: 4px;
73 }
74}
75</style>
76
use-record.ts:
1import { ref } from 'vue';
2import * as rrweb from 'rrweb';
3import rrwebPlayer from 'rrweb-player';
4import { eventWithTime } from '@rrweb/types';
5
6export const useRecord = () => {
7 const replayer = ref<HTMLElement>();
8 const showReplay = ref(false);
9 const eventList = ref<eventWithTime[]>([]);
10 const stopFn = ref();
11
12
13 const onRecord = () => {
14 stopFn.value = rrweb.record({
15 emit: event => {
16 eventList.value.push(event);
17 },
18 recordCanvas: true,
19 collectFonts: true,
20 });
21 };
22
23 const onReplay = () => {
24 stopFn.value();
25 showReplay.value = true;
26
27 setTimeout(() => {
28 new rrwebPlayer({
29 target: replayer.value as HTMLElement,
30 props: {
31 events: eventList.value,
32 },
33 });
34 }, 500);
35 };
36
37 const goBack = () => {
38 showReplay.value = false;
39 eventList.value = [];
40 };
41
42 return {
43 replayer,
44 showReplay,
45 onRecord,
46 onReplay,
47 goBack,
48 };
49};