高可用用户中心设计

概述

我们公司现在的账户体系比较混乱,每个系统都有一套自己的账户体系,没有办法进行统一的账户管理,比如统计一个人在哪个系统有账号,这样操作是比较麻烦的,还有就是我们如何进行用户行为分析,最终综合用户的信息进行用户的画像。这些都是需要解决的问题。

常规的用户设计是统一账号和统一登录验证,统一注册等信息。但是结合我们公司的实际,发现这样的话我们的系统可能会出问题,我们无法屏蔽很多细节。我们的现状是:

  • 系统需要向用户隐藏是同一个公司的这个事实。也就是说我们做了用户中心的设计,但是不能让用户感知这个系统的存在。也就是说多个系统之间的登录都是需要隔离的。离论上不能做统一账号的登录,但是可以做统一账号的验证。
  • 同一个系统可能有多个马甲app,我想马甲app之间也需要登录隔离,虽然他们调用的是同一个后台系统,那么我们不得不面对一个问题,同一个用户同时下单的逻辑处理,需要后台做这种处理,否则就会出现严重的并发错误,而开发人员可能都没有注意到这个事实的存在,从而导致一些稀奇古怪的事情出现。
  • web端与app端应该是需要同时登录的,我们好像没有web的要求,但是好像有微信的web端的事情。假如我们有web端的登录,那么问题来了,我们要不要实现二维码登录web。目前看是可以先不实现的,但是作为设计考量我们不得不考虑这些。
  • 第三方账户的登录,我们是通过手机号作为唯一标识,那么不太好实现第三方的登录注册。但是我们可以后期实现,如果实现第三方的登录注册,那么问题来了,我们的设计貌似不支持,需要改动,就是改掉手机作为唯一标识的问题,但是想想好像是不影响的,只要数据能插入问题就不大,因为没有手机号,这些用户是不能登录需要手机的app,但是有的系统可以做特殊处理,从而支持第三方账户的登录注册。但是这样做也有缺点,就是用户利用多种方式进行注册,也就是说我们不得不解决一个问题就是用户可能存在多个账户的可能性。加入用户有微信注册的账户,然后又用手机登录,绑定微信,这个时候我们可能需要去系统中用微信的unionId或者openId查找一下是否有用户,有就不能绑定,没有才能绑定。如果有我们可能也知道了这个人有重复号的可能性。这个可能需要在做的时候处理。
  • 验证保密问题,公司居然都没有人提到这点。不过也对我们是通过手机号进行验证的。但是如果以后我们不在使用手机号,可能我们需要验证问题。还需要判断用户的活动记录,从而决定用户多长时间需要重新验证app的token。比如用户换了手机,那么我们需要加强验证,不许用户名密码登陆,强制验证码登陆,同时更改登陆设备的问题。比如1年没有登陆的用户,我们是否考虑验证问题,因为可能这个号已经弃用,别人申请了这个号,那么这个新号可以冒用别人的身份进行借钱,我不知道公司现在是否有这样的处理机制,可能是出现这种事情的概率还很小,大家都不关心吧!其实就算不验证手机号,1年没操作也应该需要重新风控的吧!异常情况出现了也就是赔钱,还能干嘛。有的公司不处理应该是因为他们不是敏感行业,其实我们真的是不一样,需要更多地为用户考量这些事情,anyway扯远了。验证问题其实需要综合客服系统,还有就是自动决策也是可以的。

可以看到用户中心得设计其实牵扯的面是比较多的,其实单纯地拆分这个系统,我们发现我们的用户中心被搞重了,因为正常的用户中心不应该考虑token的sso系统,而只是信息的验证和维护提供。我们融进了SSO,还需要将数据推出验证。总之不是一件简单的事情。下面我们具体分析一下设计用户中心的思路。

用户基本信息的设计

这些可能需要根据每个公司自己的需要进行建立了,一般推荐看一下Facebook、Twitter、Linkin里面的资料详情,一般基础资料包括个人基础信息、工作信息、朋友信息、个人身份信息(护照、驾照、身份证、户口簿等)、银行卡信息等等。

信息一般登录后缓存,但是不能全部缓存,之后会提到相关的内容,还有就是实时信息需要写接口单独从数据库查。

接口的提供一般也是getSimpleInfo和getInfo两种,后者不缓存,前者一般是缓存的,所以可以再扩展就是是否需要实时信息。

关于用户体系

我们采用Account-user的账户体系设计,就是用户中心维护Account表作为基础数据或者是默认数据。每个系统可以选择维护user表,或者选择不维护。这样我们可以使有user表系统的用户使用Account表的一些数据作为默认数据,比如说头像什么的。

如果系统不使用user表,这就表示,它就是使用公共模块的数据,这时候你就发现,有user表的系统默认数据不是空的。其实这是个很有意思的设计。

再来谈谈用户名密码吧!不同的系统可能想使用不同的用户名密码登陆,然后你就会发现如果我们采用统一的Account体系是没有办法解决这个问题的,这时候user表的强大作用就凸显出来了。我们可以在user表中自己设置user:passwd字段,这样你就特例化了自己系统的账户密码,验证时你自己验证,通过后就不需要用户中心帮助你验证,你直接告诉用户中心,这个人我验证通过了,你不需要验证,你帮这个人生成token吧!你看过程简洁,职责分离,我们用户中心,就是做统一用户的可见性,聚合用户数据,但是从来不会强制你必须使用我的东西,但是你需要告知我一声就可以了,但是token必须我来统一生成。

但是大家可能会问这样token安全吗?我们的token是安全的;因为我们token的生成方式不同,token对每一个系统的可见性不同,也就是说我们的token验证是有限制的,是隔离的。

还是就是必须要提的一点就是多马甲app,不能实现单独的密码和账户,但是如果这个系统如果基于user表扩展一张验证表通过app的编号是可以解决这个问题的。这个看具体自己的系统实现吧,这个功能不归用户中心管理。

多系统用户无缝迁移

原来有那么多的老系统如何进行无缝的迁移这种平滑过渡也是我们不得不考虑的问题。一般是利用系统进行本系统的上线,但是本系统上线切入一些老系统的时候,我们又不得不面对一个问题,如何在老系统中平滑过渡?

我们慢慢发现这个其实是个bug,我们系统越多我们需要平滑迁移的次数就越多,而且要做好多我们不太情愿做的事情。因为每个系统再迁入的时候不是立马替换,而是使用两者共存的策略平滑过渡,试运行,通过之后再慢慢迁过来,一是此过程耗时间,二是需要在每个系统迁移的时候都做,加剧了迁移的时长。

还有就是系统共存时要相互导数据,这个也比较繁琐。涉及数据格式的转换,所以通用用户系统必须考虑数据的广度问题,太窄不行。还有就是一些老数据需要做一次批量导入,这个很难把握一个度,如何切分哪些数据是老的,哪些数据是MQ推来的新数据。不过我觉得先上,然后全量导入,因为导入过的数据是不会再导入的,因为有判断,在确定导入完成后,新系统再对外访问,这样就可以切数据。

迁移过程,就是一个验证,定型替换的过程,耗时费力。要进行各种评估。

token层级问题与生成策略

这里我们简单提提token隔离的问题,上面提到一个用户可能存在多个token,但是你仔细分析,可能各个Token对每个不相关子系统是不可见的。

我们是会提供一个通用的Token,这个需要用户中心的一些信息,你只要支持通用Token,那么你就和其他一些支持通用Token的系统一样,这样这些系统和标准的用户中心设计是一样的。也就是说我们的系统是支持标准的用户中心SSO Token实现方式的。

另外就是我们所说的,每个系统是可以单独生成Token的,而且生成的Token可以根据系统编号、系统分组、平台(app、web),如果是app的话,我们还会考虑马甲app的编号。

但是所有的这些不是必须的,可以让系统自己进行相应的配置的,下面我们可以进行详细的讲解。

Token的生成存放策略

我们不是讲Token的生成,关于Token的生成其实方式有很多种,sha1是我常用的,还有uuid、jwt等等各种方式,但这不是我们要讲解的重点。我们主要讲解Token怎么存放,系统如何取得Token进行交互验证。

首先系统通过传入用户名密码验证,然后我们会帮助生成Token,或者系统自行验证身份调用用户中心,请求用户中心生成Token。用户中心一般完成工作后会返回Token和这个用户的基础信息。当然注册也需要各个系统进行实现。

各个系统需要传入系统号、分组id号、是否支持web/app同时登陆、app的编号、是否分组、是否支持马甲用户隔离、是否使用公共token。

如果支持分组,也就是说系统是可以让分组内的用户共享一个Token的,这个是我们考虑到部分系统可能需要建立小全体而进行的设计。

是否支持马甲用户,是我们考虑马甲用户Token的隔离而使用的。如果为true那么我们就必须考虑为每个马甲生成不同的Token,当然前提是用户是登录了多个马甲。

==是否web、app分离,有的时候我们是允许用户web和app同时在线的,但是一般的系统都会做联动,也就是app下线了,webToken会跟着下线,这里我们可能要做很多的处理。==

是否支持通用Token,支持通用Token的话,那么我们很多事情就比较好处理,其他什么的都不用考虑了。

还有一点是要注意的,我们需要弄清楚以上参数出现的组合,有些值是可以组合的,但是有些值是不能组合的,这个需要具体的分析,==今天我就不写了,想好了再来补吧!==

那我们如何保存呢?很简单利用系统编码、分组编码、平台(web、app、client)、马甲app编码进行redis key的拼接,这个存放在redis里面,然后对应的系统也如此拼接,取得Token,并进行交互验证,这样我们的设计就完成了。

我们可能要设计jar让各个系统解耦,不需要他们写代码,我们提供相应的实现即可。这样他们只管处理自己的业务就好。

公用Token登录隐藏

下面我们要来聊一下。公共Token登录功能的设计,这里我们只会去了解上面的参数设计,然后帮助登录的时候基于这些参数做一些设计。

这里面可能只关注一个参数的设计那就是Token是否web、app分开登录的功能。

Token验证包的设计

我们需要提供一个功能的jar包,这样做是为了客户端的解耦设计。这个功能的设计涉及公共模块的处理,以及相关策略的编写。

关于Token如何验证的问题

关于token的验证,其实我们也在考虑,我们最后发现是两种方式吧,一是由sso统一验证处理,二是sso提供验证生成token,有系统自己维护token验证。

sso统一处理

这种方式看似职责比较清晰,而且其他系统与本系统的关系也是比较解耦的;但是每次系统有请求,都要把token传过来进行验证,一是用户中心的压力会变大,二是其他系统也无法做定制化的开发。

网络的依赖很重,如果多次调用用户中心,也会加大系统出错风险,如果一台机器负载很大,那对用户中心的压力也会增大,导致用户中心,对其他系统的服务能力下降,可能引起系统生态风险,当然你也可以为每个系统设置阈值,但是这个阈值需要使整个系统的负载而确定,如果其他系统请求增加,那么就降低这个系统的阈值。但是每个系统的权重又不一样,你还要设置每个系统的权重,在分配阈值的时候,你还要参考这个系统的权重,可谓不简单。

所以综上考虑我们最终放弃了这种方式的实现,转而安照下面系统的思路来进行设计。

但是网上也有使用Getway的方式进行相应地鉴权,这个就是独立地调用用户中心(SSO)进行相应的鉴权,但是这种方式我觉得会加重系统对Getway和SSO系统的依赖,加大他们的压力。同时可能也不太适合我们这种希望逻辑上实现多系统隔离的实际情况。

但是这种方式最大的好处就是职责隔离,我其实很讨厌职责不清的情况出现。还有就是改动的话,对所有系统是不可见的,这样可以无缝升级。业务系统不在关注鉴权的功能,也不需要实现登录验证的功能。

SSO验证生成Token,业务系统校验

这种设计思路是最后我们采用的设计思路,主要是用户中心负责检验用户名密码然后生成Token,存在缓存中,还要决定是否存储用户信息,然后将Token和用户信息返回给业务系统,再由业务系统进行相应地鉴权操作。

这种方式最大的好处就是减少了网络依赖的问题,同时也可每个系统高度定制化,同时也能实现用户信息的统一管理。这种特性其实特别符合我们希望我们系统能够在逻辑上相互隔离的设计。缺点就是业务系统需要做额外的处理,用户中心只是一个集中协调的角色,不太做一些其他的内容,但是这其实很符合我们现在的现状。

Token过期的问题

关于Token我们需要续期策略,如何对Token进行续期也是我们要解决的问题;其实我们是业务系统和用户系统共用token,这样的话就可以将Token的续期问题交由业务系统自己去进行处理。用户中心这边也可以感觉到在活用户数量,已经用户的Token统计信息。

用户交互分析

关于用户交互的分析,一是我们需要记录用户的交互信息,但是活跃信息由于每次Token鉴权是交给了业务系统去处理,那么我们其实并不知道用户的活跃信息。

二是用户的注册信息我们是可以进行保存,我们会记录用户手机型号,浏览器型号,ip地址,注册来源,是否有人推荐等等一系列信息。还有一点就是用户邀请功能的建立,我们的系统设计的时候貌似也没考虑到这个问题。一点一点添加吧!

三是用户登录信息的保存,这个用户中心也可以进行相应地保存。

可能还有一些其他信息的保存,这里想到了再来添加吧!另外我觉得用户的活跃信息要么走mq出来,要么打日志进行日志分析才是王道。这也是系统设计的一方面。而且这个开发过程必须加快,或者直接打入到流中,由流数据进行处理。

Token缓存策略设计

Redis并不是无限容量的。我们必须计算一个用户登录之后产生的内存。一个用户可能同时存在多个Token,有的是长时间的Token,所以我们可能维护很多Token,这样我们的Redis缓存一定是Cover不了这些信息的。所以我们必须控制Redis里面存储的用户数据库,如果超了,只能继续存到DB中,我们想采取的策略就是LRU策略,关于这种策略这里就不多介绍了。

但是其实这种实现方式也有一些问题,那就是DB频繁的硬删和插入。所以我们可以这样进行解决,DB里面存储所有人的信息。频繁去更新is_deleted字段,另外就是刷新最后交互时间。

但是又不得不面对另一个问题,那就是我们是业务系统进行鉴权,他无法操作用户中心的DB,所以用户中心这边可能需要提供一些检查接口来进行支持。总之这种方式可以防止我们系统因为Redis数据量过大而导致的系统卡顿现象的出现。

系统的高度定制化

一个高效的系统一定是可定制的,应该开启定制接口,提供定制化的参数。下面就简单分析一下吧,虽然我们不会去处理,也许系统有边界,但是思维从来没有边界,不能让自己的思维停留在系统的实现上,应该站得更高。

  • 一是我们的Token每个系统的独立是可以配置的,我们会在公共化的jar包中提供相应的定制参数。由每个业务系统定制。Token时长的配置等
  • Token存储阈值的配置,这样我们可以动态配置系统的默认行为
  • 阈值自动优化机制,这个需要系统额外信息的指引,请求量、内存等相关的数据支撑。

其实我对团队成员所说的,大概就是这些吧!

其他思考

  • 分表分库
  • 通用日志表的设计(觉得列存储,日志分析比较适合)
  • 通用id的生成策略问题

总结

用户中心设计其实关联了很多的东西,其实并没有那么好实现,我们所采用的方式其实比较新颖。估计业内很少像我这样来处理用户中心这个负载的Token生成策略与验证。我们其实还在开发阶段,但是我经过设计和各种思考觉得这个方案大体是可行的,而且我还带领团队对这个方案进行评估,大家都觉得可行。我相信团队的力量的是无限的。

总之系统的设计涉及的面其实很多东西,很多系统的设计其实是经不住推敲,一推敲你会发现一大堆的问题急需解决,但这其实并不能带来什么效益。对于团队我是希望大家牢记Google工程师的玩笑:“做正确的事,等着被开除”。但是我会继续带着团队去解决这样的问题。其实我知道团队成员对我的认可来自实力和能力,而外部成员的认可来自绩效。你觉得我会看重哪个方面。不提了,继续前行。

以后若有新的思考,就会添加新的东西。

来源:https://cordate.github.io/系统设计/高可用用户中心设计.html

评论

此博客中的热门博文

远离你身边的煤气灯人

斐讯N1小钢炮Docker安装OpenWrt/LEDE做旁路由

小程序框架全面测评

TailScale 实现远端访问整段局域网(ZeroTier另一选择)

如何去做code review

Sniper 一个轻量级 go 业务框架的思考

php照样可以用领域驱动设计DDD四层架构

大项目解决方法:代码解耦与团队解耦

N1 PT下载小钢炮固件下载及安装说明