You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

122 lines
10 KiB
Markdown

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

# 50 | 深入浅出网站高可用架构设计
你好,我是茹炳晟。今天我和你分享的主题是:深入浅出网站高可用架构设计。
在今天这篇文章中,我将沿着网站架构的话题,和你继续聊聊高可用的架构设计。
顾名思义网站高可用指的就是在绝大多的时间里网站一直处于可以对外提供服务的正常状态。业界通常使用有多少个“9”来衡量网站的可用性指标具体的计算公式也很简单就是一段时间内比如一年网站可用的时间占总时间的百分比。
我用下面这个表格,列出了四种最常见的可用性等级指标,以及允许的系统不可用时长。
![](https://static001.geekbang.org/resource/image/25/30/2510233c57cf223b83b25d9d14d60d30.jpg)
一般我们以“年”为单位来统计网站的可用性等级。“9”的个数越多一年中允许的不可用时间就越短当达到5个“9”的时候系统全年不可用时间只有区区5分钟可想而知这个指标非常难达到。
所以一般来讲业界的网站能做到4个“9”也就是说在一年内只有53分钟的时间网站是处于不可用状态就已经是算是非常优秀了。
另外可用性指标还有个特点越往后越难提高需要付出的经济成本和技术成本都会呈现类似指数级的增长。因此在实际的网站架构设计过程中到底需要做到几个“9”还需要结合具体的业务要求以及风险评估来最终确定。
那么,接下来我就首先和你分析一下造成网站不可用的主要原因,然后再基于这些原因谈谈我们可以通过哪些对策和方法,将这些造成网站不可用的因素的影响降到最低。
其实,造成网站不可用的主要原因有以下三大类:
1. 服务器硬件故障;
2. 发布新应用的过程;
3. 应用程序本身的问题。
## 服务器硬件故障
网站物理架构中,随机的硬件服务器的故障,比如某台服务器由于硬件故障宕机,可以说不是偶然,而是必然会发生的。尤其是目前互联网企业普遍采用的“牲口”模式集群方案。
而且随着网站规模不断扩大,网站后台的服务器数量也越来越多,所以由硬件故障引起问题的概率也是不断飙升。
所以,网站的高可用架构设计,需要保障的是即使出现了硬件故障,也要保证系统的高可用。
## 发布新应用的过程
网站的新版本发布过程中,往往会出现需要重新部署新的应用程序版本,然后再重启服务的情况。如果这个更新过程中不采用特殊技术手段的话,也会造成短暂的服务不可用。而且这种形式的不可用,相比服务器硬件故障的不可用更为常见。
原因很简单,互联网网站的功能更新迭代非常快,基本都是以“天”为单位来发布上线的,也就是说几乎每天都有需要中断服务来完成服务升级的可能。
显然从业务角度来看这种为了应用升级造成的服务不可用完全不可能被接受。这就好比eBay或者淘宝告诉你说我们每天某个时间段需要内部升级维护无法对外提供服务一样让人无法接受。
从网站可用性指标的角度来看,这种频繁出现的停机升级过程将大大增加网站的不可用时间。因此,我们的高可用架构设计必须能够提供切实可行的方案,将这种停机升级的影响降到最小。
## 应用程序本身的问题
造成网站不可用的最后一个原因是,应用程序本身的问题。
比如,发布的应用程序版本身存在潜在的内存泄露,那么经过较长时间的运行积累后,最终会造成服务器的内存被占满,之后必须要靠重启服务来恢复。那么,这个时候就会引入短暂的服务不可用时间。
再比如,应用程序在测试环境没有经过充分的测试验证,或者说由于测试环境的配置和实际生产环境之间存在差异,有可能造成应用程序在生产环境部署完后无法使用的情况,从而造成服务不可用。
由此可见,应用程序在上线发布前进行充分、全面的测试,是多么的重要。无论是立竿见影就能发现的功能缺陷,还是需要长期运行才能暴露的软件问题,都可以通过软件测试去发现,然后反馈给开发人员去解决,从而避免造成系统的不可用。同时,我们也需要尽可能减少测试环境和生产环境的差异,尽可能采用完全相同的环境以及第三方依赖。
## 网站高可用架构设计
为了系统性地解决造成系统不可用的上述三类问题,提高网站的可用性,我们在网站高可用架构设计上,探索出了对应的三类方法。
* 第一类方法是,从硬件层面加入必要的冗余;
* 第二类方法是,灰度发布;
* 第三类方法是,加强应用上线前的测试,或者开启预发布验证。
**对于第一类硬件故障造成的网站不可用,最直接的解决方案就是从硬件层面加入必要的冗余,同时充分发挥集群的“牲口”优势。**
比如,对于应用服务器来说,即使没有伸缩性的要求,我们也会至少采用两台同样的服务器,并且引入一台额外的负载均衡器,所有的外部请求会先到负载均衡器,然后由负载均衡器根据不同的分配算法选择其中的某一台服务器来提供服务。
> 备注:伸缩性是指通过增加或减少服务器的数量,就可以扩大或者减小网站整体处理能力。我会在下一篇文章中和你详细分享。
这样,当其中一台服务器硬件出现问题甚至宕机的时候,另一服务器可以继续对外提供服务。这时,在外部看来系统整体依然是可用的,这就给恢复那台故障服务器提供了时间。而两台服务器同时出现硬件故障的概率是很低的。
因此,从测试人员的角度来看,知道了应用服务器集群的工作原理,就可以在设计测试的时候,针对集群中的某一个或者某几个节点的故障情况设计测试用例。
再比如,对于数据存储的服务器来说,往往通过数据冗余备份和失效转移机制来实现高可用。为了防止存储数据的服务器发生硬件故障而造成数据丢失,我们往往会引入多个数据存储服务器,并且会在数据有更新操作的时候自动同步多个数据存储服务器上的数据。
也就是说,数据的存储存在多个副本,那么当某台数据存储服务器故障的时候,我们就可以快速切换到没有故障的服务器,以此保证数据存储的高可用。
那么,从测试人员的角度来看,我们依旧可以针对这种情况设计出针对部分数据服务器发生故障时的测试用例,以完成系统应对故障的反应情况的测试。
**对于第二类由于发布新应用造成的系统不可用,我们采用的主要技术手段是灰度发布。**
使用灰度发布的前提是应用服务器必须采用集群架构。假定现在有一个包含100个节点的集群需要升级安装新的应用版本那么这个时候的更新过程应该是
* 首先,从负载均衡器的服务器列表中删除其中的一个节点;
* 然后,将新版本的应用部署到这台删除的节点中并重启该服务;
* 重启完成后,将包含新版本应用的节点重新挂载到负载均衡服务器中,让其真正接受外部流量,并严密观察新版本应用的行为;
* 如果没有问题,那么将会重复以上步骤将下一个节点升级成新版本应用。如果有问题,就会回滚这个节点的上一个版本。
* 如此反复直至集群中这100个节点全部更新为新版本应用。
在这个升级的过程中,服务对外来看一直处于正常状态,宏观上并没有出现系统不可用的情况。就好比是为正在飞行中的飞机更换引擎,而飞机始终处于“正常飞行”的状态一样。
**对于第三类应用程序本身的问题造成的系统不可用,我们一方面要加强应用程序上线部署前的测试以保证应用本身的质量,另一方面需要启用所谓的预发布验证。**
我们一定遇到过这样的尴尬情况:应用在测试环境中经过了完整、全面的测试,并且所有发现的缺陷也已经被修复并验证通过了,可是一旦发布到了生产环境,还是立马暴露出了很多问题。
这其中的主要原因是,测试环境和生产环境存在差异。比如,网络环境的限制可能不一样;再比如,依赖的第三方服务也可能不一样,测试环境连接的是第三方服务的沙箱环境,而生产环境连接的是真实环境。
为了避免这类由于环境差异造成的问题我们往往会预发布服务器。预发布服务器和真实的服务所处的环境没有任何差别连接的第三方服务也没有任何差别唯一不同的是预发布服务器不会通过负载均衡服务器对外暴露只有知道其IP地址的内部人员才可以对其进行访问。
此时,我们就可以借助自动化测试来对应用做快速的验证测试。如果测试通过,新的应用版本就会进入到之前介绍的灰度发布阶段。这种做法,可以尽最大可能保证上线应用的可用性。
## 总结
今天我和你分享了衡量网站高可用性的指标对于一些大型网站来说达到4个“9”即99.99%一年中的不可用时间不超过53分钟已经算是优秀了。
然后,我将影响网站高可用的因素归为了三类,并相应地给出了解决这三类问题的方案:
1. 由服务器硬件故障引起的网站不可用,对应的解决方案是从硬件层面加入必要的冗余;
2. 由发布新应用的过程引入的网站不可用,对应的解决方案是采用灰度发布的技术手段;
3. 由应用本身质量引入的网站不可用,对应的解决方案是,一方面加强测试提高应用本身的质量,另一方面是引入预发布服务器消除测试环境和生产环境的差异。
## 思考题
关于高可用架构设计,我在文章中和你分享了应用服务器和数据存储服务器的高可用架构。但是,我并没有介绍缓存服务器的高可用架构。那么,你认为缓存服务器是否也需要高可用架构的支持呢?如果需要的话,缓存集群的高可用架构应当如何设计?如果不需要,也请你分享一下你的理由。
感谢你的收听,欢迎你给我留言。