[译]揭秘英雄联盟的数据服务器

译自:Riot Games Engineering

杰微刊刘晓鹏翻译:揭秘英雄联盟的数据服务器

Hey,大家好!我是 Bill “LtRandolph” Clark,一名英雄联盟的游戏工程师。许多 Rioter 工程师关注大量的内容需要直接发送给玩家问题——这是两个我最近最喜欢的例子之一,包括最新的冠军Jhin及项目重构的支持。而我的团队使得这个过程变得又快有简单。

我们有一个简单的目标:即允许参加游戏试玩项目的暴民,能够创建两倍于任何给定的LoL(英雄联盟)补丁的内容。这说起来容易,但是执行起来却是一个充满挑战的任务。

今天,我们讨论实现这一目标我们所铺设的基础:Riot 游戏数据服务器(GDS)。虽然这是一篇技术文章,但是我会站在一个较高的层次来解释这个问题。如果你是一个在做多系统间传送数据工作的工程师,我希望这能让你特别感兴趣。

游戏数据:

首先,我们了解一些背景。在LoL的工作中,存在两种类型的游戏数据:一种是 key-value 对,被称为属性数据(如 Black Cleaver HP 奖金是300),另一种是不透明的二进制数据(如,大文本、动画和视频)。在这篇文章中,我们只讨论属性数据,二进制数据处理是未来潜在的一篇博文。

在LoL的所有历史中,属性数据由一堆松散、混乱的文件组成,这些文件存储在一个大的名为 DATA 的文件夹中。

早期,我们将数据存储在.ini的文件中(对,就是 Windows 下 .ini的文件格式)。类似如下所示:


没有漂亮的界面


当然,我创建这个例子是为了强调一些我们在编辑.ini文件时遇到的一些共同的问题。这离用户友的界面相差甚远。编辑原始文本时非常容易混乱——缺乏重要的内容,而其他字段又重复。设计者们每天不得不处理这种混乱,这里总共有 977 种法术,这些功能(当然忽略)位于“MissileEffect=AnnieBasicAttack_mis.troy”行中,在很早的LoL开发中,每个冠军涉及一个令人愉快的场景:“Death=Cardmaster_Death.wav。”

下面是当前数据系统面临的一些关键问题:

1、使用 Notepad++ 来编辑属性数据

2、对已存在的字段没有清晰的定义

3、缺乏类型安全

4、多人同时编辑同一个文件时会有合并冲突问题

5、繁琐的并行版本(活跃(Live)版、公测版(PBE)及内部版本)

6、松散的文件链接;只有短名称和隐藏的搜索路径

游戏数据服务器

我们特意设计一个游戏数据服务器系统来定位这些问题。最基础的是 RiotGameDataServer.exe——一段运行在每个开发者机器上的小程序。它以一个 Riot 拳头形式显示在工具栏,它所做的工作是连接属性数据与计算机上的程序


准备着的GDS

GDS 为其他工具抽取出文件和数据的管理方式,所以这些工具只需要关注传送过来数据的展示和编辑。我将其类比于操作系统窗口创建的抽象来思考,所以一个开发者只需要关注窗口该如何显示即可。GDS 工具还包括许多内部开发工具,也有第三方工具,如 Maya 和 Photoshop。它们都通过基于 JSON 格式数据的 RPC API 与 GDS 进行通信。

关于一份整洁的 RPC API,我们可以很容易的通过使用一个名为Swagger的标准来生成一个文档页,它列出了所有的有效函数。这是 GDS 暴露的一小部分函数子集:

Swagger 文档

GDS 属性数据存在一个名叫 PROPERTIES 的文件夹中。该文件夹最终将包含所有英雄联盟的属性数据。当一个工具需要识别出在什么情况下 Black Cleaver 是 Pantheon 最喜欢的武器时,它就会给 localhost:1300(GDS 监听的端口)发一个 HTTP 请求。当接收到一个“get?path=Items/BlackCleaver”请求时,GDS 就会去 PROPERTIES/Items/BlackCleaver.json文件中查找。响应结果类似如下所示:


稍微漂亮一点

当某个工具想要改变 Black Cleaver 造成的伤害值时,该工具需要发送另一条命令到localhost(或 127.0.0.1)的 1300 端口上,这次需要发送的指令是“et&path=PROPERTIES/Items/BlackCleaver.FlatHPMod&value=1000.”。GDS 工具将从源码控制工具(Perforce)中获取到指定的文件,并编辑对应的值,然后将成功或失败的结果返回给页面。因此,任何工具都可以很简单的修改属性数据文件,而不需要考虑数据的格式,文件操作或者其他复杂的因素。

这样,我们就很容易创建工具,如RiotEditor,来解决问题 #1:使用 Notepad++ 来编辑属性数据。


现在,我们总算有点结果了

属性标记

对任何给定的类型,非常重要的一点是识别出其实际在的字段,这样我们才能知道用户可以编辑什么。为了完成这项工作,我们维护了一个环环相扣的宏与magic 模板集合,该集合允许我们在工程代码里面直接标注类型。大概形式如下所示:


宏的修改

注意宏:PROPERTY_CLASS,PROPERTY_START,PROPERTY及 PROPERTY_END。它们负责两项主要任务:

1、告知类定义了哪些出口,类的哪些字段应该是可编辑的。

2、告知属性加载系统内存的偏移量,以便在运行时载入属性值。

PROPERTY 宏可通过特定的简单模板函数自动推断类型。我们可以引用复杂类型,如BoundingVolume,只需要提供它们拥有的子属性的标注。我们也可以跳过某些字段,如mRuntimeNumber,这意味着它们不会在 GDS 中暴露出来。

这是 GDS 中使用的 JSON 定义的结果:


Aww yiss

属性标记解决了问题 #2 和 #3:分别是对已存在的字段没有清晰的定义和没有类型安全。

GDS 除了为其他工具抽取出文件和数据的管理方式外,还做了一件相当酷的事情,就是为 Riot 开发工程师提供一项技术,我们称之为“层”。一层代表了一个可以关闭或打开的功能,我们可以为一个新的冠军,一种新皮肤,一个游戏模型或一次大的重新平衡创建一个层。然后,当一个内容创建者在某一功能上工作时,他们可以告诉 GDS,例如,“激活”APItemRework层。

之后,GDS 会对APItemRework层包含的文件任何改变打上标签。在磁盘上,这看起来像一个文件,我们称之为RabadonsDeathcap.json,挨着该文件的另一个文件,名为RabadonsDeathcap.APItemRework.json。在第二个层文件中,GDS 简单的标记每个被改变的字段为 delta。保存前后两个值就是为了解决之后的合并冲突。这两个文件并排看起来如下所示:


左侧:基础数据;右侧:delta 层

由于我们捕获了单个字段的改变,我们不再需要担心多个 Rioters 同时修改同一个文件了,除非他们修改完全相同的字段。如果他们修改了同一个字段,我们也存储了修改前后的值,所以可以识别出冲突。这样做的好处是可以防止一个发包以后的bug:在创建 DJ Sona,团队意外的将其状态恢复到了上一个版本。

现在,我们解决了问题 #4:多人同时编辑同一个文件时的合并冲突问题。

分层,让我们捕获了某一特定功能的所有改变。为了真正发布这些功能,我们需要引入一个概念,叫做“游戏版本”。一个游戏版本定义了一个完整的打开层的集合。每个游戏版本保存了一个层名称的简单 JSON 列表。在任何给定的时间点,我们维护几个主要的游戏版本:

1、Alpha:内部测试、准备发布到公测服务器上的功能集合。

2、Beta:当前公测服务器上的功能集合,如Jhin。要注意的是,Beta 版继承了 Release 版的功能列表,所以它拥有最近更新的功能,类似于季前赛。

3、Release:当前正式服务器上的功能集合,如闪亮新补丁包 6.3。

还有一个很酷的特性是一个功能从一个版本迁移到另一版本只需要在我们层管理窗口执行一次拖拽操作即可,有了这个以后,我们不再需要在某个功能发生改变时,需将成百上千个文件从一个地方拖到另一个地方了。

这样就很容易解决了问题 #5:全文件覆盖的并发版本(活跃版、公测版及内部版本)

总结:

希望这篇文章能让你体会到我们是如何使得LoL开发更有效率的。对于细心的读者,你可能发现我们没有深入讨论问题 #6:松散的文件链接;只有短名称和隐藏的搜索路径。我留下这个问题没解决是因为这个问题比预期的更麻烦——存在冗余、避免不必要的代码重构、增加数据迁移、补丁大小等问题,因此,值得专门为此写一篇完整的博客。

如果你对我们如何改变英雄联盟中混乱的游戏数据的某方面感兴趣,请务必在评论中告知我们。

更多精彩译文

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容