Files

529 lines
11 KiB
Vue
Raw Permalink Normal View History

<!--
IncidentResponseDemo.vue
故障响应流程演示展示从故障发现到复盘的完整流程
-->
<template>
<div class="incident-demo">
<div class="header">
<div class="title">
故障响应流程 (Incident Response)
</div>
<div class="subtitle">
专业团队如何处理线上故障
</div>
</div>
<div class="timeline">
<div
v-for="(phase, index) in phases"
:key="phase.id"
:class="[
'phase',
{ active: activePhase === index, completed: activePhase > index }
]"
@click="activePhase = index"
>
<div class="phase-marker">
{{ index + 1 }}
</div>
<div class="phase-content">
<div class="phase-title">
{{ phase.title }}
</div>
<div class="phase-time">
{{ phase.time }}
</div>
<div class="phase-desc">
{{ phase.desc }}
</div>
<div
v-if="activePhase === index"
class="phase-actions"
>
<div class="action-title">
关键动作
</div>
<ul class="action-list">
<li
v-for="action in phase.actions"
:key="action"
>
{{ action }}
</li>
</ul>
<div
v-if="phase.tools"
class="tools-section"
>
<div class="tools-title">
常用工具
</div>
<div class="tools-list">
<span
v-for="tool in phase.tools"
:key="tool"
class="tool-tag"
>{{ tool }}</span>
</div>
</div>
</div>
</div>
</div>
</div>
<div
v-if="activePhase === phases.length - 1"
class="incident-meta"
>
<div class="meta-title">
📋 故障复盘报告 (Post-mortem)
</div>
<div class="meta-content">
<div class="meta-section">
<div class="meta-label">
故障等级
</div>
<div class="meta-value level-p1">
P1 - 高优先级
</div>
</div>
<div class="meta-section">
<div class="meta-label">
影响范围
</div>
<div class="meta-value">
15% 用户无法访问订单服务
</div>
</div>
<div class="meta-section">
<div class="meta-label">
故障时长
</div>
<div class="meta-value">
23 分钟
</div>
</div>
<div class="meta-section">
<div class="meta-label">
根本原因
</div>
<div class="meta-value">
数据库连接池配置过小高峰期连接耗尽
</div>
</div>
<div class="meta-section">
<div class="meta-label">
改进措施
</div>
<div class="meta-value">
1. 增加连接池大小至 200
<br>
2. 添加连接池监控告警
<br>
3. 优化慢查询减少连接占用时间
</div>
</div>
</div>
</div>
<div class="best-practices">
<div class="practice-title">
🎯 故障处理最佳实践
</div>
<div class="practice-grid">
<div class="practice-card">
<div class="practice-icon">
</div>
<div class="practice-name">
快速响应
</div>
<div class="practice-desc">
建立 15 分钟响应机制P0 故障立即电话通知
</div>
</div>
<div class="practice-card">
<div class="practice-icon">
📢
</div>
<div class="practice-name">
信息同步
</div>
<div class="practice-desc">
定期向用户和内部同步故障进展避免猜测
</div>
</div>
<div class="practice-card">
<div class="practice-icon">
🔍
</div>
<div class="practice-name">
保留现场
</div>
<div class="practice-desc">
故障现场数据日志监控完整留存便于分析
</div>
</div>
<div class="practice-card">
<div class="practice-icon">
📝
</div>
<div class="practice-name">
blameless 文化
</div>
<div class="practice-desc">
复盘对事不对人聚焦流程改进而非个人责任
</div>
</div>
</div>
</div>
</div>
</template>
<script setup>
import { ref } from 'vue'
const activePhase = ref(0)
const phases = [
{
id: 'detect',
title: '故障发现',
time: 'T+0 分钟',
desc: '监控系统自动发现异常指标',
actions: [
'监控检测到订单服务错误率从 0.1% 飙升到 8.5%',
'Alertmanager 立即触发 P1 告警',
'值班人员收到钉钉和短信通知'
],
tools: ['Prometheus', 'Grafana', 'Alertmanager']
},
{
id: 'respond',
title: '快速响应',
time: 'T+3 分钟',
desc: '值班人员确认故障并启动应急流程',
actions: [
'登录监控面板确认故障范围',
'创建线上故障 War Room 会议',
'通知相关开发人员和运维人员'
],
tools: ['钉钉/飞书', 'Zoom/腾讯会议']
},
{
id: 'diagnose',
title: '故障定位',
time: 'T+8 分钟',
desc: '通过日志和追踪系统分析根因',
actions: [
'查看应用日志,发现大量 "Connection pool exhausted" 错误',
'通过链路追踪定位到数据库查询耗时异常',
'检查数据库监控,发现连接池已满'
],
tools: ['ELK', 'Jaeger/Zipkin', 'Arthas', 'tcpdump']
},
{
id: 'fix',
title: '故障修复',
time: 'T+18 分钟',
desc: '实施临时解决方案恢复服务',
actions: [
'紧急扩容数据库连接池从 50 到 200',
'重启应用服务使配置生效',
'监控显示错误率逐渐下降到正常水平'
],
tools: ['K8s Dashboard', 'kubectl', 'ansible']
},
{
id: 'verify',
title: '恢复验证',
time: 'T+21 分钟',
desc: '确认服务完全恢复正常',
actions: [
'监控指标全部回到正常范围',
'执行冒烟测试验证核心功能',
'观察 5 分钟无异常,宣布故障结束'
],
tools: ['Postman', '自动化测试平台']
},
{
id: 'postmortem',
title: '故障复盘',
time: 'T+48 小时',
desc: '总结经验教训,制定改进计划',
actions: [
'召开复盘会议,整理故障时间线',
'编写 Post-mortem 报告',
'跟进改进措施落实情况'
],
tools: ['Confluence/Notion', 'JIRA']
}
]
</script>
<style scoped>
.incident-demo {
border: 1px solid var(--vp-c-divider);
background: var(--vp-c-bg-soft);
border-radius: 12px;
padding: 1.5rem;
margin: 1.5rem 0;
font-family: var(--vp-font-family-base);
}
.header {
margin-bottom: 1.5rem;
}
.title {
font-weight: 700;
font-size: 1.1rem;
margin-bottom: 0.25rem;
}
.subtitle {
color: var(--vp-c-text-2);
font-size: 0.9rem;
}
.timeline {
display: flex;
flex-direction: column;
gap: 1rem;
margin-bottom: 1.5rem;
}
.phase {
display: flex;
gap: 1rem;
padding: 0.75rem;
background: var(--vp-c-bg);
border-radius: 10px;
border: 2px solid var(--vp-c-divider);
cursor: pointer;
transition: all 0.3s;
}
.phase:hover {
border-color: var(--vp-c-brand);
}
.phase.active {
border-color: var(--vp-c-brand);
background: rgba(var(--vp-c-brand-rgb), 0.05);
}
.phase.completed {
border-color: #22c55e;
background: rgba(34, 197, 94, 0.02);
}
.phase-marker {
width: 40px;
height: 40px;
border-radius: 50%;
background: var(--vp-c-bg-soft);
border: 2px solid var(--vp-c-divider);
display: flex;
align-items: center;
justify-content: center;
font-weight: 700;
font-size: 1rem;
flex-shrink: 0;
}
.phase.active .phase-marker {
border-color: var(--vp-c-brand);
color: var(--vp-c-brand);
background: rgba(var(--vp-c-brand-rgb), 0.1);
}
.phase.completed .phase-marker {
border-color: #22c55e;
color: #22c55e;
background: rgba(34, 197, 94, 0.1);
}
.phase-content {
flex: 1;
}
.phase-title {
font-weight: 700;
font-size: 1rem;
margin-bottom: 0.25rem;
}
.phase-time {
font-size: 0.85rem;
color: var(--vp-c-brand);
font-weight: 600;
margin-bottom: 0.5rem;
}
.phase-desc {
font-size: 0.9rem;
color: var(--vp-c-text-2);
margin-bottom: 0.75rem;
}
.phase-actions {
padding: 0.75rem;
background: var(--vp-c-bg-soft);
border-radius: 6px;
margin-top: 0.75rem;
}
.action-title {
font-weight: 600;
font-size: 0.85rem;
margin-bottom: 0.5rem;
}
.action-list {
margin: 0;
padding-left: 1.25rem;
font-size: 0.85rem;
color: var(--vp-c-text-2);
}
.action-list li {
margin-bottom: 0.25rem;
}
.tools-section {
margin-top: 0.75rem;
}
.tools-title {
font-weight: 600;
font-size: 0.85rem;
margin-bottom: 0.5rem;
}
.tools-list {
display: flex;
gap: 0.5rem;
flex-wrap: wrap;
}
.tool-tag {
padding: 0.25rem 0.5rem;
background: var(--vp-c-bg);
border: 1px solid var(--vp-c-divider);
border-radius: 4px;
font-size: 0.75rem;
font-family: var(--vp-font-family-mono);
}
.incident-meta {
background: var(--vp-c-bg);
border-radius: 10px;
padding: 0.75rem;
margin-bottom: 1.5rem;
border: 1px solid var(--vp-c-divider);
}
.meta-title {
font-weight: 700;
font-size: 1rem;
margin-bottom: 1rem;
}
.meta-content {
display: flex;
flex-direction: column;
gap: 0.75rem;
}
.meta-section {
display: flex;
gap: 1rem;
font-size: 0.9rem;
}
.meta-label {
min-width: 100px;
font-weight: 600;
color: var(--vp-c-text-1);
}
.meta-value {
flex: 1;
color: var(--vp-c-text-2);
line-height: 1.6;
}
.meta-value.level-p1 {
color: #f59e0b;
font-weight: 700;
}
.best-practices {
background: rgba(var(--vp-c-brand-rgb), 0.05);
border-radius: 10px;
padding: 0.75rem;
border: 1px solid var(--vp-c-brand);
}
.practice-title {
font-weight: 700;
font-size: 1rem;
margin-bottom: 1rem;
}
.practice-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
gap: 1rem;
}
.practice-card {
background: var(--vp-c-bg);
border-radius: 6px;
padding: 0.75rem;
text-align: center;
}
.practice-icon {
font-size: 2rem;
margin-bottom: 0.5rem;
}
.practice-name {
font-weight: 700;
font-size: 0.9rem;
margin-bottom: 0.5rem;
}
.practice-desc {
font-size: 0.8rem;
color: var(--vp-c-text-2);
line-height: 1.5;
}
@media (max-width: 768px) {
.phase {
flex-direction: column;
}
.phase-marker {
width: 32px;
height: 32px;
}
.meta-section {
flex-direction: column;
gap: 0.25rem;
}
.meta-label {
min-width: auto;
}
.practice-grid {
grid-template-columns: 1fr;
}
}
</style>