秒送LBS场景下的C端SOA服务容灾建设之-数据备份篇

电子说

1.3w人已加入

描述

前言

在面向服务的架构(SOA)系统中,容灾能力是保障系统稳定性的重要组成部分。通过引入多数据中心部署、自动化故障转移、数据备份等技术手段,可以有效提升系统在面对突发灾难事件时的恢复能力。例如,采用主从复制和异地多活架构,可以确保在某个数据中心发生故障时,其他数据中心能够迅速接管业务,避免服务中断。此外,定期进行灾难恢复演练和系统压力测试,也是提高容灾能力的关键措施。通过这些手段,SOA系统能够在各种极端情况下保持高可用性和稳定性,从而为用户提供持续可靠的服务。今天重点聊一下LBS场景下的数据备份方案。

SOA

一、问题背景

随着秒送业务的上线,基于LBS的用户交易的流量逐渐提升,不同于JD主站的B2C场景,O2O场景下的资源场景划分的非常精细,根据门店履约围栏划分3公里、2公里、1公里等不同的范围,资源与资源之间互相交错、覆盖,构建成一个具有LBS特性的资源网。作为入口级流量的后台SOA服务,我们需要根据业务特性区分哪些数据强实时/弱实时来制定不同的数据备份策略、还需要弄清楚底层数据生产的逻辑链路,才能保障数据备份的真实性、还需要考虑用户使用的这份备份数据体验等等因素。因此,LBS场景下的容灾数据备份的难度瞬间升级。

二、痛点/挑战

基于上述背景描述,我们在制定方案的过程中遇到了很多挑战,主要分为三大类

1、如何缓存poi经纬度数据

如果要按照全国poi经纬度去缓存数据,为了降低数据差异,我们把不同poi点的距离精确到(米级),那么全国poi的数量是多少呢?根据主站网格地址统计,精确度在75米的poi点在6.8亿+,如果按照1米算,估计千亿+级别的poi。如此量级的poi数据,我们如果每一个poi都做数据备份的话(理想状态);拿秒送首页、频道页举例,首页一次交互数据大小600kb左右,频道页核心5个,每个400kb左右,那么每个poi需要2600KB的存储,按照75米的poi距离,需要6.8亿*2600kb=1646TB,C端数据最终存储在redis中,按照5G=1核=15元/月,那么需要花费500万+人民币/月,老板听完方案汇报脸都黑了。

2、如何降低容灾数据存储成本

每个月500多万的存储开销成本显然是不现实的,于是我们迎来了第二个挑战,那就是怎么才能降低这么大的存储成本,于是我们内部多次讨论,绞尽脑汁,得出结论:如果想让用户得到最好的体验那么似乎和成本之间是互斥的,我们怎么才能寻找一个中间点,来平衡户体验和成本。

3、如何保障缓存数据资源的有效性

在进行数据备份时,数据一致性问题是不可避免的关键点。考虑到如此庞大的数据量,若采用实时请求备份的方式,SOA层面的请求负载将可能对底层RPC造成极大的查询压力。相对而言,异步构建方法则要求我们复制主站搜索、推荐和GIS定位等异构逻辑(因为秒送首页、门店详情等资源来源于主站搜索、推荐和GIS),这不仅意味着需要处理上千个中台的消息队列(MQ),还要确保逻辑与现有系统保持一致,这显然是个巨大挑战。

因此,我们的选择余地有限。有观点认为,数据备份并不需要完全真实,稍有偏差也是可接受的,甚至可以使用静态图片代替。我们承认,实现数据容灾备份与线上数据保持100%一致是不现实的。但我们的目标是尽可能保证90%的用户体验与线上相似,同时确保交易流程的连续性,并在系统异常时为用户提供无感知的体验(此处价值观强调升华,用户为先)

三、业界方案调研

flag立完了,但是确实没啥好思路,B2C场景的不适用,只能是去跟隔壁同行取取经,通过多方渠道找到了一些靠谱的某团的技术大佬,并且做了1对1的沟通。大致梳理了下他们的架构和容灾方案。

某团的超市便利架构:

SOA

通过沟通得知,某团在SOA层,没有做任何的数据备份容灾,完全依赖底层的数据提供,如果底层数据返回空,那么页面也会空白...至于底层数据有没有做容灾数据备份,就不得而知,我相信是有的。于是到这里,感觉前路又被堵死了。

四、方案构思

调研无果的情况下,我们回过头来重新整理下现有的逻辑链路,从中寻找一些突破点,为了整体容灾数据备份方案做预设计。

1、首先根据业务筛选下,哪些入口是阻塞交易链路,不可降级的

从中识别出:首页、频道页、门详页是阻塞交易的环节,需要进行数据备份。

2、底层数据依赖的链路逻辑分别是什么

(1)首页、频道页

首页和频道页充当着网站的主入口,它们利用CMS系统和定向投放系统实现了基于地理位置服务(LBS)的楼层内容过滤。网站内容通过商品组合、广告组合以及推荐机制构建,核心资源主要依赖于推荐系统的数据池。我们分析后注意到,不同用户在同一地点或同一用户在不同地点可见的内容各不相同,这主要是由推荐系统的数据池结合特定策略实现的。推荐数据所依赖的因素来源于多种策略的分配,而这些策略因素又会反过来影响推荐结果。这导致我们无法仅通过备份当前数据来保持数据的一致性,因为策略的即时变化可能会使备份数据的结果发生变化。若要备份数据与线上数据完全一致,我们需要复制一整套推荐系统的数据流转过程,这是目前的一个难点。为了应对这种情况,我们采用了一个通用账号执行普通策略来进行数据备份,以尽可能排除变量因素,虽然这牺牲了一定的个性化推荐能力,但确保了数据正常展现。

SOA

(2)门详页

门详页是秒送流量的关键交易通道,其核心门店分类和商品信息均通过搜索引擎和商家分类中台获得。尽管集团的主搜索引擎已经分离出专门的垂直搜索服务供秒送场景使用,但数据源仍然是共用的,区别仅在于网关层面的分离。与首页和频道页不同,门详页专注于门店层面的分类和商品数据,不需涉及广告投放和目标人群等因素。只要门店能在首页展示,用户点击进入后看到的分类和商品就会保持一致。因此,我们的主要关注点是确保当前搜索引擎的数据源是准确的。我们对线上100家不同类型的门店(包括连锁品牌和非连锁品牌)进行了一周的观察,发现门店分类变动较少,而分类下的商品状态和库存变化较频繁。基于此,我们可以对门店分类实施全量缓存(考虑到较低的时效性需求),而对于分类下的商品则根据情况实施缓存(需要较高的时效性)。

SOA

3、削减数据备份量级思路

因为底层依赖的数据源不一致,所以首页、门详页分别制定不同的数据备份设计方案,首页的由于和poi强相关,所以更复杂一些。门详因为不涉及poi以及策略,所以相对首页来说容易一些。

(1)首页、频道页(引入网格思想)

在整理推荐系统的数据链路时,我们注意到推荐结果中的门店商品资源高度依赖GIS系统。深入分析GIS逻辑后,了解到GIS维护着不同覆盖范围的网格围栏,如3公里、5公里和20公里网格,并且每个网格下的门店商品资源是相同的。这启发了我们,如果我们也能构建一个类似GIS的网格系统并缓存门店商品,岂不是能显著提升数据的一致性?然而,现实告诉我们,推荐系统调用GIS接口是基于策略模型的选择,并非易事。若要完整复制推荐系统逻辑几乎不可能。

然而,网格的概念是值得借鉴的,因为它可以大幅减少我们的POI缓存数据规模。如果我们反其道而行之,直接缓存当前POI下的推荐结果,就无需处理推荐逻辑的复杂性。接下来的挑战是构建一套尽可能与GIS网格重叠的网格系统。

重点来了!

如何选择构建网格类型?

开源的网格grid构建空间索引又很多种,墨卡托、geohash、H3、S2等等,那么我们如何去选择呢?GIS提供的网格接口使用的空间索引是geohash四边形网格,但是为了以防万一,我们通过主站网格中台封装的JMF组件,异构了包含了geohash在内的多种空间索引,目的是全方位覆盖,为了后续的diff提供多种选择,得到最优结果。这里我就挑一种H3作为demo去讲述如何构建的网格。

SOA

如何去构建网格?

另外一个问题,构建网格的分辨率应该如何选择,也就是我们常说的网格精度。我们去门店中台查了秒送的门店范围,发现90%以上的门店都是基于3公里圈定的不规则履约范围,而这个也是作为GIS判断网格是否覆盖该门店的核心因子。所以在构建网格的时候,每个蜂窝格子的面积限定3公里显然不能得到最优结果,于是我们通过H3的空间索引构建了 Hexagon六边形网格,尽可能的模拟线上真实不规则的履约圈选范围。同时Hexagon的边长精度划定为7也就是1.4km,得到最小六边形面积3.12m²,最大六边形面积6.22m²。同时针对精度8的也异构出一套数据作为diff参照。经过计算后得出(刨去偏远地区),精度7的热点网格数量在几十万+,精度8个网格数量在百万+,和原本的百亿级别相比,降低了99%~~,最后经过成本计算:①精度7的=2973元/月 ;②精度8的=14133元/月 (相比于最初的500万+/月,打折到骨折!)

Hexagon官方文档:

SOA

如何识别网格热点poi?

(选择网格中的哪个poi作为当前网格的缓存代表?)

构建好网格后,问题又来了,因为网格是一组poi的集合,我们只有拿到一个网格内部数据备份最好的poi经纬度,才能去模拟线上用户请求进行缓存,作为这个网格的容灾数据备份,从而达到最优的数据备份效果。经过和网格中台沟通后得出,网格集成了格子内的用户请求分布情况,于是我们可以根据用户分布最集中的经纬度poi点进行数据备份,达到最佳效果。

SOA


 

数据备份时间多久?

最后就是数据备份的时间,也就是数据一致性的问题。上面说过了,推荐包含了很多策略,同一个用户 白天和晚上看到的首页结果都不一样,这个就是涉及到体验和成本的问题,如果为了保持好的体验,我们可以缓存2套数据,白天一套、晚上一套。但是考虑到晚上用户占比过少,一期还是优先缓存白天的数据。

(2)门详页

门详页的容灾备份上面提到了,我们只需要考虑如何缓存门店分类数据和商品数据,分类数据一个门店最多几百个,但是商品可能往往上万,掏出小本本,我们再算一笔账:我们现在百万+门店,每个门店的商品乘以每条商品数据平均按1kb算,那么成本预估15W元+/月,领导听完汇报脸又黑了。

策略选择

我们经过讨论后,决定先对每个末级分类的前两页商品进行缓存,并且剔除掉闭店的门店。经过策略估算后,缓存量级可以压缩到900元/月

(3)引入GIZP压缩

经过实践后发现,gizp压缩对于纯文本的压缩预估能节省60%的空间,但是需要注意的是QUERY数据解压缩的性能耗时。我们在进行设计的时候,把所有的数据全部转化成String字符串,并且通过gizp压缩。最终得到以下备份数据量级:


 
首页 频道页 门详页
压缩前 1087GB 3624GB 300GB
压缩后 652GB 2174GB 180GB
最终成本 1956元/月 6522元/月 540元/月

至此,整个容灾数据备份量级压缩方案可行,相比于最初的500W+/月,降低到了9018元/月。(还要啥自行车?领导当即拍板,赶紧做完上线)

4、DIFF思路-如何去和线上做DIFF?

(1)首页、频道页

到了验证环节了,因为我们无法完全和GIS的网格重合,那么我们需要一套规则来做DIFF验证网格的热点poi数据备份是否和线上poi数据返回一致(不可能100%一致,但是需要知道是否达到90%以上的效果),我们选择六边形网格的六个定点poi以及中心poi,共计7个poi去请求线上,并且用返回的数据和热点poi请求返回的数据做DIFF,核心39城每个城市抽查10组,最终得到一个比例。然后通过不同的空间索引精度,去挨个验证出一个最佳的索引精度。

SOA

(2)门详页

门详DIFF验证,抽选出重点KA门店,定时任务check门店下的分类和商品情况。如果抽查整体DIFF异常率超过30%,我们则会触发新一轮的异构任务。

SOA

五、落地过程

上面的方案构思经过无数次的推演以及前期验证确保了方案的可行性。那么接下来就是到了落地环节,在这个环节里,我们发现其实客户端本身也有缓存,但是缓存的大小非常有限。客户端采用的localStorage本地缓存存储,是基于用户的手机内存做的,5M左右的大小空间。所以适合做一些兜底灾备的缓存,比如服务端彻底down的情况下,应急去读客户手机内存缓存。用户每次成功请求服务端后,都会重写更新缓存,于是,一个可行(邪恶


审核编辑 黄宇

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分