Cloudflare 在七月二日发生了一次全球性的宕机,个人托管在上面的两个小网站难以幸免,502 超过半个小时;甚至上班的时候,还收到了一些业务告警(某些渠道通过 cloudflare 做路由)。可见这次故障的影响范围之大,互联网的一些基础服务已经成为了 21 世纪的水电煤..
而作为一名 SRE,明白在故障的整个生命周期中,最关键的一环就是故障复盘(postmortem),以防止同样愚蠢的错误不再发生(通常大故障都是由很多小错误连锁造成的)。前天在千岛湖 outing 半夜四点睡不着的时候,起床偶遇这篇文章《Details of the Cloudflare outage on July 2, 2019》, 一口气读完了,写的很精彩(很会讲故事),当然总觉得还缺了什么。
用这篇文章记录一下个人的感受和思考,当然更加推荐阅读原文。
什么是 Cloudflare ?
今天刷 Twitter 的时候刚好看到一个官方的回答还挺不错的。总结一下原理就是在用户与你的网站之间加了一层代理,以提升 security, performance and reliability.
故障过程
按原文描述整理后,习惯将故障的每一步都按 timeline 列出来(UTC):
故障原因
代码变更::更新 WAF 规则时,引入一个很容易回溯 (backtrace) 的正则表达式,尝试用画图去解释 backtrace 的原理(总步数与原文不一致,是因为我省略了一些步骤方便理解):(Why is zero plural?🤔😄)
当然如果你不想细看,也可以直接看原文中的动画,有个大致的体感:
总而言之,就是随着字符串的增加,正则匹配的时间复杂度爆炸增长,最终导致 CPU 资源耗尽:
故障根因
原文中列了很多,我挑出了个人认为导致故障最重要的三点:
- WAF 规则变动直接自动部署到生产环境: cloudflare 的代码变更,正常的部署模式为DOG(内部员工) → PIG(部分免费用户) → Canaries(灰度) → Global(全量部署), 但对于 WAF 规则的变更,由于常常需要快速全球部署以便于快速应急,所以日常的模式也都是无脑一把推。
- 全球流量下降的警报没有第一时间发出
- 回滚预案执行超时: 回滚需要编译两遍代码 + 权限与内部平台失效的问题,导致无法第一时间止血。
针对上面的两点思考:
- 控制变更风险: 绝大部分的故障都是由变更导致的,所以在我们公司有一个任何人都不能触碰的红线叫做变更三板斧:对于任意线上变更都需要 可灰度,可监控,可回滚。如果 cloudflare 在这次故障中的变更中,遵守了三板斧的任意一个,都会大大减少故障恢复的时间,甚至避免故障。变更三板斧真的是无数人血与泪传承下来的 golden rules.
- 保鲜:不管是 Netflix 的 chaosmonkey 还是蚂蚁的红蓝攻防,都是一种比较好的业内最佳实践了,只有真实频繁的去模拟故障,才能做到预案与应急能力的保鲜 & 提升。
其他思考
"This generated a Change Request ticket. We use Jira to manage these tickets and a screenshot is below."
"In the last 60 days, 476 change requests have been handled for the WAF Managed Rules (averaging one every 3 hours)."
Cloudflare 的变更管理&感知看上去做的挺不错的, 但真正发生故障的时候,也花了将近 18 分钟才定位到根因。公司也有很多人在做「变更感知」与「故障定位」,但个人觉得或许相对于感知,故障根因定位(例如定位关联的变更)才是最难的:比较依赖于专家经验,但真正大故障又往往是之前从来没有遇到过的场景。所以是不是换个思路,对与所有的跌零因子都做好预案,故障发生时,只需要无脑执行预案即可,第一时间止血恢复业务,之后慢慢排查原因。
"I agree 100% transparency is best. "
"blame free".
国外和国内文化上比较大的两个差别,但仁者见仁智者见智,大环境的限制,也很难说有所谓的对错吧。
链接参考
- 《Details of the Cloudflare outage on July 2, 2019》: https://blog.cloudflare.com/details-of-the-cloudflare-outage-on-july-2-2019/
- pageduty: https://www.pagerduty.com/
彩蛋
最后的最后,这句话真的很触动我。。
This is the first thing I've ever done professionally that I truly, completely love. I wake up every day just thrilled at the work we're doing. Source
--EOF--