首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

我们怎样把一个数百万用户的网站从REST API迁移到GraphQL API?

关于从 REST API 迁移到 GraphQL API 的好处,已经有很多人写过这方面的文章。假设你已经接受了这个事实,那么如果你想要转换一个拥有数百万用户的网站,确保性能不受影响,并且真的不想把它搞砸,你该怎么做?

我们从去年开始了这个旅程,并获得成功,现在我们就回过头来讲这个事。我们的 GraphQL API 现在是 OkCupid 的官方 API,被所有的客户端调用,包括我们的 iOS 和 Android App,以及我们的桌面和移动 Web 单页 React 应用程序。

这篇文章将讲述我们是如何处理这个巨大的项目的。我将介绍我们所构建的东西、我们为测试即将发布的新代码而制定的策略,以及一些在技术方面本可以做得更好的地方。

注:本文更多地是关于过程而不是代码本身,我们将在另一篇文章中讨论与性能有关的问题。

收益

我们的 GraphQL API 已经在生产环境中运行了一年半。在一年多前,我们就停止向 REST API 添加新特性了。GraphQL API 由 227 个实体组成,每分钟处理 17 万个请求。

我们还没有完全弃用 REST API,从请求量方面来看,客户端的迁移工作进行了一半多,但从实体数量来看,可能还不到一半。

我们是怎么做的

对我们来说,这些是一个全新的技术栈和代码库(Node、Apollo Server、Docker),所以我们需要制定一个计划,在不中断生产环境的情况下验证其有效性。我们的流程是:

  1. 选择一个适当的页面进行转换;
  2. 构建schema;
  3. 添加影子请求来调用新的API,同时仍然通过REST API获取数据;
  4. 使用真实用户进行A/B测试。

我们在 2019 年 1 月开始这个项目,在 1 月 28 日发布影子查询,在 3 月 13 日开始 A/B 测试,并在 4 月 30 日发布完整版。所以,经过 4 个“简单”的步骤,你也可以在 4 个月内得到一个在生产环境中运行的 GraphQL API。

接下来,让我们深入探究每一个步骤。

选择一个适当的页面进行转换

我们决定将 OkCupid 对话页面作为切入点。在这个页面中,用户可以看到自己正在进行的对话,也可以看到“匹配”列表(可以开始新对话的人):

转换之前的对话页面

选择一个网站核心页面是很重要的,这样可以帮你确定好约定条件,充实数据模型的重要部分,为未来的工作打好基础,并更好地进行概念验证。页面越“真实”,就越能帮你了解新 API 是否可行。

我们选择了对话页面,需要考虑如何呈现它:

  • User:用户基本信息;
  • Match:两个用户相互关联的状态信息(例如,匹配百分比,一个用户是否喜欢另一个用户,等等);
  • Conversation:基本的对话信息(例如,发送者、最后一条消息的片段、发送时间,等等)。

我们还要考虑到一些可重用的 API 概念,比如分页。

构建 schema

对很多第一次进行 schema 设计的团队来说,这可能是一个具有挑战性的步骤——对我来说就是如此!这里有一些建议:

  • 调研。与schema有关的文章有很多,例如GraphQL文档提供的基本示例、GitHub和Yelp的公共API、Relay的文档等等。在这里要感谢Apollo团队,他们为我们提供了大量帮助。
  • 不要担心REST API的数据格式问题,最好将schema设计得更具表达性和惯用性,不要受旧API的限制。
  • 保持一致。我们的旧API主要使用了蛇形命名格式,但也存在一些难看的组合词(例如userid和displayname)。这时候刚好是纠正这些字段名称的好机会!
  • 具体化。GraphQL API中的字段名称越具体,在进行重大更改时,就越容易迁移成新字段。例如,User.essaysWithDefaultsUser.essays更好。
  • 基于调研结果为团队做一些有用的东西。例如,在研究分页标准时,我本来想用Relay的规范,但发现它对edgenode等术语太过依赖,对于客户端来说不够友好(我们最终决定返回一个data列表)。

加入影子请求

在 GraphQL 为真实用户提供数据前,我们在生产环境中使用影子请求对系统进行测试:在我们的目标页面上,用户向 REST API 请求数据,在显示 REST 数据之后,再向 GraphQL API 发出相同的请求。这样我们就可以比较两个 API 的性能,并在用户发现问题前修复它们。

我们当然不是第一批想到这么做的人,但这对我们来说是非常重要的一步。

我们在这个 API 的第一个草案上花费的时间几乎是 REST API 的两倍,这显然不是很酷。使用影子请求让我们可以在不影响实际用户体验的情况下诊断性能问题。

进行实验

最后一步是使用真实用户来测试新的 API 。因为已经验证了响应时间与影子请求是差不多的,所以我们有信心进行 A/B 测试。

如果你期望在实验中看不到变化,那么这种实验是没有意义的,因为你试图证明什么都没有发生。在这样的实验中,你关心的统计数据在本质上是没有意义的,除非发生了什么问题。

因此,你应该为实验设置一个持续时间,而不是观察统计数据是否发生了显著变化。一旦达到了设定的持续时间,并且仍然没有看到显著的变化,就可以对系统满怀信心了。

在我们的案例中,这个持续时间是一个月(每组实验超过了 10 万用户)。

哪些地方可以做得更好?

初稿总是不完美的(即使是第二稿也是,至少对我来说)。虽然发布 API 的过程进行得很顺利,但在发布之后我们还是学到了一些技术上的东西。

错误处理

我们没有针对 GraphQL 更新 API 返回的错误定义好结构。当我们意识到这个问题时,已经向客户端显示了各种各样的错误。一个看起来比较好的解决方案是标准化一个 Error 类型,我们可以在给定的消息体中对其进行扩展。这篇文章非常深入地讲解了如何设计错误类型。

业务逻辑应该放在哪里?

当遇到涉及业务规则的功能时,你会很容易想到把逻辑添加到 API 层,特别是如果你需要依赖另一个团队来实现它们。

例如,我们开发了一个显示所有喜欢你并给你发信息的人的功能。我们向付费用户显示整个列表,但对于免费用户,只显示第一项,然后是一系列占位符。这个功能的第一个版本在 API 层有检查用户付费状态并用占位符替换列表项的逻辑。

在使用 GraphQL API 一段时间后,我们意识到把业务逻辑集中在后端是最好的,而 GraphQL API 的作用是以一种对客户端来说有意义的方式获取、格式化和显示后端的数据。

总结

总的来说,我们的流程是可行的。我们可以快速地将一些东西发布到生产环境中,以此来验证我们的技术决策是否可行,在错误对用户造成影响前修复它们,并针对旧 API 来测试新的变更。

如果你决定进行类似的迁移,希望这个流程能够给你带来帮助。

原文链接:

https://tech.okcupid.com/moving-okcupid-from-rest-to-graphql/

  • 发表于:
  • 本文为 InfoQ 中文站特供稿件
  • 首发地址https://www.infoq.cn/article/J5ssBqFvgiexs44wNkpQ
  • 如有侵权,请联系 cloudcommunity@tencent.com 删除。

扫码

添加站长 进交流群

领取专属 10元无门槛券

私享最新 技术干货

扫码加入开发者社群
领券