- 原文地址 https://blog.poetries.top/2022/07/27/sentry-summary/
- 前端面试之旅:https://interview2.poetries.top
- 公众号:「前端进阶之旅」 每天分享技术干货
Sentry介绍
Sentry
是一套开源的实时的异常收集、追踪、监控系统。这套解决方案由对应各种语言的 SDK 和一套庞大的数据后台服务组成,通过 Sentry SDK 的配置,还可以上报错误关联的版本信息、发布环境。同时 Sentry SDK 会自动捕捉异常发生前的相关操作,便于后续异常追踪。异常数据上报到数据服务之后,会通过过滤、关键信息提取、归纳展示在数据后台的 Web 界面中
- Github: https://github.com/getsentry/sentry
- 文档 https://docs.sentry.io/
支持如下语言
data:image/s3,"s3://crabby-images/ebf82/ebf82b3b9db25f7890666b39cfeecc6d2d908a9b" alt=""
sentry功能架构
data:image/s3,"s3://crabby-images/94c0d/94c0d28525cee673dfa895d419cf54bdcc5c1893" alt=""
sentry核心架构
data:image/s3,"s3://crabby-images/15e06/15e069ea6e98fdac4a62e819a2f17fbeeb87a7da" alt=""
环境搭建配置
官方Sentry服务
sentry是开源的,如果我们愿意付费的话,sentry给我们提供了方便。省去了自己搭建和维护 Python 服务的麻烦事
登录官网 https://sentry.io 注册账号后接入sdk即可使用
data:image/s3,"s3://crabby-images/9583f/9583f0026eeebe4c9a92f7ca8a118d6579aa6119" alt=""
Sentry私有化部署
Sentry 的管理后台是基于
Python Django
开发的。这个管理后台由背后的 Postgres 数据库(管理后台默认的数据库)、ClickHouse
(存数据特征的数据库)、relay、kafka、redis 等一些基础服务或由 Sentry 官方维护的总共 23 个服务支撑运行。可见的是,如果独立的部署和维护这 23 个服务将是异常复杂和困难的。幸运的是,官方提供了基于 docker 镜像的一键部署实现 getsentry/onpremise
sentry 本身是基于 Django 开发的,而且也依赖到其他的如 Postgresql、 Redis 等组件,所以一般有两种途径进行安装:通过 Docker 或用 Python 搭建
前置环境
需要安装对应版本,否则安装会报错
Docker 19.03.6+
Docker-Compose 1.28.0+
4 CPU Cores
8 GB RAM
20 GB Free Disk Space
安装docker环境
安装工具包
yum install yum-utils device-mapper-persistent-data lvm2 -y
data:image/s3,"s3://crabby-images/bb131/bb1315e9d64e6be7f8b77cea3c93cf722a21598e" alt=""
设置阿里镜像源
yum-config-manager --add-repo https://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo
data:image/s3,"s3://crabby-images/ca27c/ca27c2637e06c029f1629599c0dd0995827721f5" alt=""
安装docker
yum install docker-ce docker-ce-cli containerd.io -y
启动docker
systemctl start docker
# 设为开机启动
systemctl enable docker
docker 镜像加速(重要)
在后续部署的过程中,需要拉取大量镜像,官方源拉取较慢,可以修改 docker 镜像源
登录阿里云官网,打开 阿里云容器镜像服务。点击左侧菜单最下面的 镜像加速器
,选择 Centos
data:image/s3,"s3://crabby-images/8f19f/8f19f9d5e7d0e0d58b2f36b2b232cba2ba87df64" alt=""
vi /etc/docker/daemon.json
{
"registry-mirrors": ["https://l6of9ya6.mirror.aliyuncs.com"]
}
然后重启docker即可
# 重新加载配置
systemctl daemon-reload
# 重启docker
systemctl restart docker
安装docker-compose
# 使用国内源安装
sudo curl -L "https://get.daocloud.io/docker/compose/releases/download/1.29.2/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/docker-compose
设置docker-compose执行权限
chmod +x /usr/local/bin/docker-compose
创建软链
sudo ln -s /usr/local/bin/docker-compose /usr/bin/docker-compose
测试是否安装成功:
$ docker-compose --version
docker-compose version 1.22.0, build f46880fe
一键部署
git clone https://github.com/getsentry/onpremise
在
onpremise
的根路径下有一个install.sh
文件,只需要执行此脚本即可完成快速部署,脚本运行的过程中,大致会经历以下步骤:
- 环境检查
- 生成服务配置
-
docker volume
数据卷创建(可理解为docker
运行的应用的数据存储路径的创建) - 拉取和升级基础镜像
- 构建镜像
- 服务初始化
- 设置管理员账号(如果跳过此步,可手动创建)
cd onpremise
# 直接运行 ./install.sh 将 Sentry 及其依赖都通过 docker 安装
./install.sh
后续一步一步安装下来
data:image/s3,"s3://crabby-images/88357/883579c4ac55282175868d509481dc20f827f7fa" alt=""
设置管理员账号(如果跳过此步,可手动创建)
data:image/s3,"s3://crabby-images/88486/884864b4a6a0f0abe6d7cd348834f4dc7c58bf21" alt=""
启动项目执行
在执行结束后,会提示创建完毕,运行 docker-compose up -d
启动服务
data:image/s3,"s3://crabby-images/ec649/ec6497700caa3343e743b47210fb372307cf784c" alt=""
docker-compose up -d
data:image/s3,"s3://crabby-images/fe3d1/fe3d1dde1c676d685c4c4f8f5148f9a46e3c64b1" alt=""
查看服务运行状态docker-compose ps
data:image/s3,"s3://crabby-images/2da32/2da32cb9e8bde52147383effef6aa4f75429b98a" alt=""
访问项目
所有服务都启动成功后,就可以访问
sentry
后台了,后台默认运行在服务器的9000
端口,这里的账户密码就是安装时让你设置
的那个
data:image/s3,"s3://crabby-images/2d722/2d722ae5292c27e8f20a1ac9835a9ed99ecb1212" alt=""
data:image/s3,"s3://crabby-images/eb754/eb754ec89b3d12766a39a99bc6a1d42dac14e45d" alt=""
设置语言和时区
点击头像User settings - Account Details
的相应菜单设置,刷新后生效
data:image/s3,"s3://crabby-images/86b11/86b1185aa91cf16c1938dbdd77f4252c3a28c54e" alt=""
Vue2 + Sentry
创建一个vue项目
npm i @vue/cli -g
# 初始化vue2项目
vue create vue2-sentry
接入sentry
data:image/s3,"s3://crabby-images/485b3/485b313f215b9a93fb6f3a194d96975cfcfb64ed" alt=""
# Using npm
npm install --save @sentry/vue @sentry/tracing
// src/main.js
import Vue from "vue";
import Router from "vue-router";
import * as Sentry from "@sentry/vue";
import { Integrations } from "@sentry/tracing";
Vue.use(Router);
const router = new Router({
// ...
});
Sentry.init({
Vue,
dsn: "http://xdsdfafda21212@119.75.24.41:9000/2",
integrations: [
new Integrations.BrowserTracing({
routingInstrumentation: Sentry.vueRouterInstrumentation(router),
tracingOrigins: ["localhost", "my-site-url.com", /^//],
}),
],
// 不同的环境上报到不同的 environment 分类
// environment: process.env.ENVIRONMENT,
// Set tracesSampleRate to 1.0 to capture 100%
// of transactions for performance monitoring.
// We recommend adjusting this value in production
// 高访问量应用可以控制上报百分比
tracesSampleRate: 1.0,
release: process.env.SENTRY_VERSION || '0.0.1', // 版本号,每次都npm run build上传都修改版本号
});
// ...
new Vue({
router,
render: h => h(App),
}).$mount("#app");
我们手动抛出异常,在控制台可见捕获了错误
data:image/s3,"s3://crabby-images/c0261/c02613b167eaa195748ca8a259efe78bd3cef7ca" alt=""
data:image/s3,"s3://crabby-images/9e870/9e8701ee1dbff316fc7b01f6f338b52426f250a1" alt=""
上传sourceMap到sentry
为了方便查看具体的报错内容,我们需要上传sourceMap
到sentry
平台。一般有两种方式 sentry-cli
和 sentry-webpack-plugin
方式,这里为了方便采用webpack
方式
-
source-map
是一个文件,可以让已编译过的代码可以映射出原始源; -
source-map
就是一个信息文件,里面储存着位置信息。也就是说,转换后的代码的每一个位置,所对应的转换前的位置。
webpack方式上传
npm i @sentry/webpack-plugin -D
修改vue.config.js
配置文件
// vue.config.js
const SentryCliPlugin = require('@sentry/webpack-plugin')
module.exports = {
// 打包生成sourcemap,打包完上传到sentry之后在删除,不要把sourcemao传到生产环境
productionSourceMap: process.env.NODE_ENV !== 'development',
configureWebpack: config=> {
if (process.env.NODE_ENV !== 'development') {
config.plugins.push(
new SentryCliPlugin({
include: './dist/js', // 只上传js
ignore: ['node_modules', 'webpack.config.js'],
ignoreFile: '.sentrycliignore',
release: process.env.SENTRY_VERSION || '0.0.1', // 版本号,每次都npm run build上传都修改版本号 对应main.js中设置的Sentry.init版本号
cleanArtifacts: true, // Remove all the artifacts in the release before the upload.
// URL prefix to add to the beginning of all filenames. Defaults to ~/ but you might want to set this to the full URL. This is also useful if your files are stored in a sub folder. eg: url-prefix '~/static/js'
urlPrefix: '~/js', // 线上对应的url资源的相对路径 注意修改这里,否则上传sourcemap还原错误信息有问题
// urlPrefix: 关于这个,是要看你线上项目的资源地址,比如
// 怎么看资源地址呢, 例如谷歌浏览器, F12控制台, 或者去Application里面找到对应资源打开
}),
)
}
},
}
获取TOKEN
data:image/s3,"s3://crabby-images/fa118/fa1181a08769c6e9d53e4772d860ae6a6853641f" alt=""
data:image/s3,"s3://crabby-images/06de1/06de1bb7b4523d84c3149cb3b93bb7976f91890b" alt=""
data:image/s3,"s3://crabby-images/b8ea1/b8ea1f1bbf017ac65f9ba572789c057b8da6a8d9" alt=""
获取org
data:image/s3,"s3://crabby-images/1437b/1437b8a52e89ece1d6b795f457072568e1a3ca44" alt=""
在项目根目录创建.sentryclirc
-
url
:sentry部署的地址,默认是https://sentry.io/
-
org
:控制台查看组织名称 -
project
:项目名称 -
token
:生成token需要勾选project:write
项目写入权限
# .sentryclirc
[auth]
token=填入控制台创建的TOKEN
[defaults]
url=https://sentry.io/
org=sentry
project=vue
执行项目打包命令,即可把js下的sourcemap
相关文件上传到sentry
npm run build
上传后的sourcemap
在这里可以看到
data:image/s3,"s3://crabby-images/7012e/7012e03405e308dff647723889f12883422eee00" alt=""
正确上传过 source-map
的项目,可以看到很清晰的报错位置
进入本地打包的dist,
http-server -p 6002
启动一个模拟正式环境部署的服务访问看看效果
data:image/s3,"s3://crabby-images/d1c25/d1c25514f8590ef9266d86bfe77ca083f90b826d" alt=""
还可以通过 面包屑
功能查看,报错前发生了什么操作
data:image/s3,"s3://crabby-images/d4ad0/d4ad0e1ea9eba2036bde26567f4d26508e1c4ff2" alt=""
记得别把sourcemap文件传到生产环境,又大又不安全 删除sourcemap
, 基于vue2演示的三种方式
// 方式1
"scripts": {
"build": "vue-cli-service build && rimraf ./dist/js/*.map"
}
// 方式2 单独生成map
// vue.config.js
configureWebpack(config) {
config.output.sourceMapFilename('sourceMap/[name].[chunkhash].map.js')
config.plugin('sentry').use(SentryCliPlugin, [{
include: './dist/sourceMap', // 只上传js
ignore: ['node_modules'],
configFile: 'sentry.properties',
release: process.env.SENTRY_VERSION || '0.0.1', // 版本号,每次都npm run build上传都修改版本号
cleanArtifacts: true, // 先清理再上传
}])
}
// 方式3 webpack插件清理
$ npm i webpack-delete-sourcemaps-plugin -D
// vue.config.js
const { DeleteSourceMapsPlugin } = require('webpack-delete-sourcemaps-plugin')
configureWebpack(config) {
config.plugin.push(
new DeleteSourceMapsPlugin(), // 清理sourcemap
)
}
查看 Performance
data:image/s3,"s3://crabby-images/a5061/a50614c86a8a37ed5f6b0d7642453b0bcaf8cce4" alt=""
Sentry.init()
中,new Integrations.BrowserTracing()
的功能是将浏览器页面加载和导航检测作为事物,并捕获请求,指标和错误。
-
TPM
: 每分钟事务数 -
FCP
:首次内容绘制(浏览器第第一次开始渲染 dom 的时间点) -
LCP
:最大内容渲染,代表viewpoint
中最大页面元素的加载时间 -
FID
:用户首次输入延迟,可以衡量用户首次与网站交互的时间 -
CLS
:累计布局偏移,一个元素初始时和消失前的数据 -
TTFB
:首字节时间,测量用户浏览器接收页面的第一个字节的时间(可以判断缓慢来自网络请求还是页面加载问题) -
USER
:uv
数字 -
USER MISERY
: 对响应时间难以忍受的用户指标,由sentry
计算出来,阈值可以动态修改
Vue3 + Vite + Sentry
创建vue3项目
yarn create vite
安装sentry依赖
npm i @sentry/vue @sentry/tracing
初始化sentry
src/main.js
中修改
import { createApp } from "vue";
import { createRouter } from "vue-router";
import * as Sentry from "@sentry/vue";
import { Integrations } from "@sentry/tracing";
const app = createApp({
// ...
});
const router = createRouter({
// ...
});
Sentry.init({
app,
dsn: "http://xdfada1212@12.715.204.41:9000/2",
integrations: [
new Integrations.BrowserTracing({
routingInstrumentation: Sentry.vueRouterInstrumentation(router),
tracingOrigins: ["localhost", "my-site-url.com", /^//],
}),
],
// 不同的环境上报到不同的 environment 分类
// environment: process.env.ENVIRONMENT,
// Set tracesSampleRate to 1.0 to capture 100%
// of transactions for performance monitoring.
// We recommend adjusting this value in production
// 高访问量应用可以控制上报百分比
tracesSampleRate: 1.0,
release: process.env.SENTRY_VERSION || '0.0.1', // 版本号,每次都npm run build上传都修改版本号
});
app.use(router);
app.mount("#app");
sourcemap上传
修改
vite.config.js
配置
安装npm i vite-plugin-sentry -D
插件
import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'
import viteSentry from 'vite-plugin-sentry'
const sentryConfig = {
configFile: './.sentryclirc',
release: process.env.SENTRY_VERSION || '0.0.1', // 版本号,每次都npm run build上传都修改版本号
deploy: {
env: 'production',
},
skipEnvironmentCheck: true, // 可以跳过环境检查
sourceMaps: {
include: ['./dist/assets'],
ignore: ['node_modules'],
urlPrefix: '~/assets', // 注意这里设置正确,否则sourcemap上传不正确
},
}
// https://vitejs.dev/config/
export default defineConfig({
plugins: [
vue(),
process.env.NODE_ENV === 'production' ? viteSentry(sentryConfig) : null,
],
build: {
sourcemap: process.env.NODE_ENV === 'production',
},
})
此时当执行
vite build
时,viteSentry
这个插件会将构建的sourcemap
文件上传到sentry
对应的项目release
之下。当次版本新捕获到错误时就可以还原出错误行,以及详细的错误信息。
data:image/s3,"s3://crabby-images/0eda8/0eda8adde1a84a0e2e1562c1adb9497b4c96cd40" alt=""
React + Sentry
使用umi项目接入演示
创建一个umi项目
mkdir umi-sentry && cd umi-sentry
yarn create umi
data:image/s3,"s3://crabby-images/bdb63/bdb63451a92a20ccedebf06cf6fce9adf60bbce3" alt=""
# Using npm
npm install --save @sentry/react @sentry/tracing
接入sentry
初始化sentry
// pages/index.ts
import * as Sentry from "@sentry/react";
import { BrowserTracing } from "@sentry/tracing";
Sentry.init({
dsn: "https://xdfa@o1334810.ingest.sentry.io/121",
integrations: [new BrowserTracing()],
// Set tracesSampleRate to 1.0 to capture 100%
// of transactions for performance monitoring.
// We recommend adjusting this value in production
release: '0.0.1',
tracesSampleRate: 1.0,
});
手动抛出异常查看是否能正确上报到sentry
data:image/s3,"s3://crabby-images/6028d/6028da5806e577ebc43ad782146f9e7d84064527" alt=""
sourcemap上传
根目录创建配置文件 .sentryclirc
[auth]
token=TOKEN控制台获取,TOKEN需要勾选project:write写入权限
[defaults]
url=https://sentry.io/ // 默认地址
org=组织名称,控制台获取
project=react // 项目名称
sourcemap配置上传
npm i @sentry/webpack-plugin -D
// .umirc.ts 修改
const SentryPlugin = require('@sentry/webpack-plugin');
export default {
devtool: process.env.NODE_ENV === 'production' ? 'source-map' : 'eval', // 开启sourcemao
chainWebpack(config, { webpack }){
if (process.env.NODE_ENV === 'production'){//当为prod时候才进行sourcemap的上传,如果不判断,在项目运行的打包也会上传
config.plugin("sentry").use(SentryPlugin, [{
ignore: ['node_modules'],
include: './dist', //上传dist文件的js
configFile: './sentryclirc', //配置文件地址,这个一定要有,踩坑在这里,忘了写导致一直无法实现上传sourcemap
release:'1.0.1', //版本号,自己定义的变量,整个版本号在项目里面一定要对应
deleteAfterCompile: true,
urlPrefix: '~/' // js的代码路径前缀
}])
}
},
};
# 执行打包上传sourcemap
npm run build
# 进入dist文件,启动http-server 本地服务模拟线上效果
data:image/s3,"s3://crabby-images/7ef83/7ef83c3e7a6f04db2a05b8c0e9b13dab8b289fc2" alt=""
修改代码抛出异常,查看控制台sourcemap解析的效果
data:image/s3,"s3://crabby-images/048cc/048ccbdeb22824fdcb22b0b3951a22d7c9565512" alt=""
注意:npm run build之后,不要把sourcemap上传到生产环境,记得删除
进阶用法
识别用户
在上传的 issues 里面,我们可以借助 setUser 方法,设置读取存在本地的用户信息。(此信息需要持久化存储,否则刷新会消失)
// app/main.js
Sentry.setUser({
id: 'dfar12e31', // userId cookie.get('userId')
email: 'test@qq.com', // cookie.get('email')
username: 'poetry', // cookie.get('username')
})
Vue.prototype.$Sentry = Sentry
data:image/s3,"s3://crabby-images/2866b/2866b5b57ea2e3a54c28db732ad2f3f574d8da3a" alt=""
错误边界
- 定义错误边界,当组件报错的时候,可以上报相关信息
- 使用
Sentry.ErrorBoundary
。加了错误边界,可以把错误定位到组件上面。
rrweb 重播
npm i @sentry/rrweb rrweb -S
import SentryRRWeb from '@sentry/rrweb';
// app/main.js
Sentry.init({
Vue,
dsn: "xxx",
integrations: [
new Integrations.BrowserTracing({
routingInstrumentation: Sentry.vueRouterInstrumentation(router),
tracingOrigins: ["localhost", "my-site-url.com", /^//],
}),
new SentryRRWeb({
checkoutEveryNms: 10 * 1000, // 每10秒重新制作快照
checkoutEveryNth: 200, // 每 200 个 event 重新制作快照
maskAllInputs: false, // 将所有输入内容记录为 *
}),
],
// 不同的环境上报到不同的 environment 分类
environment: process.env.ENVIRONMENT,
// 高访问量应用可以控制上报百分比
tracesSampleRate: 1.0,
release: process.env.SENTRY_VERSION || '0.0.1', // 版本号,每次都npm run build上传都修改版本号
logErrors: true
});
在报错后,可以录屏播放错误发生的情况
data:image/s3,"s3://crabby-images/7b69c/7b69c0cc6e58e815d1cc1631f633c61b1ef63339" alt=""
手动设置报警
- 设置报警规则,当我们某些情况,如
issues
,performance
超过我们设置的阈值,会触发alert
。 - 我们可以通过提醒等功能来帮助我们即时发现问题。
data:image/s3,"s3://crabby-images/35ed0/35ed0505728e859df82a8fe8951a4cc155306579" alt=""