Giter Club home page Giter Club logo

iohao / iogame Goto Github PK

View Code? Open in Web Editor NEW
782.0 21.0 174.0 3.48 MB

无锁异步化、事件驱动架构设计的 java netty 网络编程框架; 轻量级,无需依赖任何第三方中间件或数据库就能支持集群、分布式; 适用于网络游戏服务器、物联网、内部系统及各种需要长连接的场景; 通过 ioGame 你可以很容易的搭建出一个集群无中心节点、集群自动化、分布式的网络服务器;FXGL、Unity、UE、Cocos Creator、Godot、Netty、Protobuf、webSocket、tcp、socket;java Netty 游戏服务器框架;

Home Page: http://game.iohao.com

License: GNU Affero General Public License v3.0

Java 100.00%
bolt game gameframework java netty springboot unity3d unreal-engine websocket framework

iogame's Introduction

ioGame

无锁异步化、事件驱动的架构设计;轻量级,无需依赖任何第三方中间件或数据库就能支持集群、分布式
通过 ioGame 可以很容易的搭建出一个集群无中心节点、集群自动化、多进程的分布式游戏服务器
包体小、启动快、内存占用少、更加的节约、无需配置文件、提供了优雅的路由访问权限控制
可同时支持多种连接方式:WS、UDP、TCP...等;框架已支持全链路调用日志跟踪特性
让开发者用一套业务代码,能轻松切换和扩展不同的通信协议:Protobuf、JSON
近原生的性能;业务框架在单线程中平均每秒可以执行 1152 万次业务逻辑
代码即联调文档、JSR380验证、断言 + 异常机制 = 更少的维护成本
框架具备智能的同进程亲和性;开发中,业务代码可定位与跳转
架构部署灵活性与多样性:既可相互独立,又可相互融合
可同时与同类型的多个游戏逻辑服通信并得到数据
逻辑服之间可相互跨进程、跨机器进行通信
支持玩家对游戏逻辑服进行动态绑定
能与任何其他框架做融合共存
对 webMVC 开发者友好
无 spring 强依赖
零学习成本
javaSE

http://game.iohao.com

JDK 21
AGPL3.0
gitee star github star



过去、现在、将来都不会有商业版本,所有功能全部开源!

只做真的完全式开源,拒绝虚假开源,售卖商业版,不搞短暂维护!

承诺项目的维护周期是十年起步, 2022-03-01起,至少十年维护期!

提供高质量的使用文档!


ioGame 源码完全开放、最新文档阅读完全开放;使用完全自由、免费(遵守开源协议)。


ioGame 是一个轻量级的网络编程框架,适用于网络游戏服务器、物联网、内部系统及各种需要长连接的场景。


使用 ioGame,可以显著的帮助企业减少巨额成本。在文档中,"成本"关键字提到了很多次,各个阶段均有关联,包括了学习、研发、测试、部署、扩展、投入 ...等各阶段。在同等资源的竞争下,使用 ioGame 可为公司节省更多的资源,从而提高了自身的生存率。更重要的是避免了为其他公司做嫁衣的可能性,具体可阅读成本分析案例


为什么采用授权许可申请?

授权许可申请 -- 点我展开
ioGame 使用的是 AGPL3.0 开源协议。
由于 ioGame 遵循过去、现在、将来都不会有商业版本,所有功能全部开源的原则,使得我们很难找到盈利点,现在使用授权的方式;

使用 ioGame 是免费的,前提是要认同生产资料公有制,就是把你的产品也完全的开源出来(遵守 AGPL3.0 开源协议);同时,ioGame 是允许私有的,前提是支付闭源授权费。所以,付费不是必须的,也不是强制性的,选择权在使用者手上。

ioGame 支持申请盈利后支付闭源授权费,收入不超过10万人民币/年或拥有不超过100万人民币的资产,可以申请盈利后支付授权费用。

### 闭源授权说明
1.授权费用:2799 人民币每年。
2.一个产品只对应一个授权。
3.取得授权的产品只可自己运营或使用,如果是提供给客户二次开发或销售类的产品,每份需要取得一个授权。

如果你的产品不想开源,可以向我们申请为期一年的闭源授权,企业用户需要通过企业邮箱申请闭源授权。如果是个人开发者、独立开发者的,可以使用非企业邮箱申请闭源授权。

### 授权类型
个人和独立开发者:收入不超过10万人民币/年或拥有不超过100万人民币的资产,可以申请盈利后支付授权费用。
企业:收入不超过10万人民币/年或拥有不超过100万人民币的资产,可以申请盈利后支付授权费用。

我们最终是期望你的产品盈利后,支付闭源授权费用。我们的目的也很明确,通过授权的方式获得资金,在使用这些资金来加快发展框架的功能及建立完善社区,更好的为大家服务。

以下行业或产品需要请提前沟通:基于 ioGame 的商业化视频、围绕 ioGame 的商业化售卖产品; 围绕 ioGame 的商业化产品指的是售卖相关的商业产品;

存在法律纠纷阶段是不予授权的。

### 申请授权
申请授权邮箱(发到 [email protected][email protected]),内容格式如下
公司全名:
统一社会信用代码:
产品描述:
软件著作权编号:(没有可以不用填写)
申请的授权类型:(个人和独立开发者、企业、专业版、定制版)


### 合法授权码
授权码是根据 【公司统一社会信用代码、软著码、发放日期、失效日期】等信息生成的,确保唯一性与合法性;

授权码会通过邮箱发放,用于留存,具备法律效力。
产品在取得授权期内,不需要对外开放产品源代码;如果不在授权期内,你的产品将转为 AGPL3.0 开源协议。



启动展示

ioGame 在内存占用、启动速度、打包等方面也是优秀的。

  • 内存方面:内存占用小。
  • 启动速度方面:应用通常会在 0.x 秒内完成启动。
  • 打包方面:打 jar 包后大约 15MB ,详细请看 快速从零编写服务器完整示例


源码、示例、效率

github gitee
ioGame - 源码 ioGame - 源码
ioGame 示例集合 ioGame 示例集合

示例

示例 描述
ioGame 综合示例介绍 示例中有功能特性的实践、打包部署(docker、k8s)等介绍
[示例] FXGL 连接示例;Protobuf、java、Netty FXGL + ioGame 网络游戏中的多人移动演示
[示例] Unity 连接示例 - 1;Protobuf、C#、Netty unity3d 连接示例 websocket + protobuf(已经与综合示例调通)
[示例] Unity 连接示例 - 2;Protobuf、C#、Netty 提供了 unity 与 ioGame 的【多人】移动同步演示
[示例] Cocos Creator 连接示例;Protobuf、TypeScript、Netty cocosCreator 连接示例 websocket + protobuf(已经与综合示例调通)
[示例] Godot 连接示例;Protobuf、C#、Netty 网络通信使用 webSocket
[示例] UE5 连接示例;Protobuf、C++、Netty UE5 连接示例 websocket + protobuf(已经与综合示例调通)
[示例] JavaScript 连接示例;json、JavaScript、Netty 使用 websocket.js 来连接 ioGame 的一个示例,使用 json 协议来传输交互。
[示例] TypeScript 连接示例;json、TypeScript、Netty 使用 json 协议来传输交互。

效率

效率
ioGame 快速理解篇 快速掌握 ioGame 的概念
技术栈推荐与其它参考文档 游戏开发的技术栈建议,大家可以参考一下。
更多游戏服务器框架推荐 如果 ioGame 不能很好的满足你的需要,请别担心,这里还有更多的 java 游戏服务器框架推荐给你。
ioGame java SDK ioGame java SDK 主要作用是与 ioGame 服务器进行通信,为开发者简化交互过程。

最小依赖

ioGame 已经上传到**仓库,如果无法下载最新的框架源码,建议开发者的 maven 仓库代理使用原生的或腾讯云的代理,目前不推荐阿里云的代理。腾讯云代理设置可参考这里

ioGame最新版本查看 https://www.yuque.com/iohao/game/ab15oe

ioGame 是轻量级的网络编程框架,不依赖任何第三方中间件或数据库就能支持集群、分布式,只需要 java 环境就可以运行。此时,你只需一个依赖即可获得整个框架,并同时支持开头介绍的全部功能特性。

<!-- https://mvnrepository.com/artifact/com.iohao.game/run-one-netty -->
<dependency>
    <groupId>com.iohao.game</groupId>
    <artifactId>run-one-netty</artifactId>
    <version>21.9</version>
</dependency>

ioGame 使用趋势数据

关注 ioGame 的游戏服务器开发者持续增多,2022-09 ~ 至今各月的统计数据;

这里的统计信息是关于开发者关注 ioGame 框架相关的,从统计数据中可以看出,由于 ioGame 上手简单,功能强大等优点,得到了众多开发者的关注。如果你想知道 ioGame 有没有人在使用,可以先到这里看下统计数据、开发者的评价与讨论。

https://www.yuque.com/iohao/game/gpxk93#TwVa8

这里展示了每月的统计数据,统计数据来源于语雀后台,这些数据都是真实的、客观存在的、活的

因为成本的原因,某宝某多还没有出现能提供这种服务的商家,所以这样的统计数据也更具真实性。

通过统计数据,我们可以看到每天会有很多开发者在访问 ioGame 的在线文档,并且这些统计数据不是来源于口嗨的,也不是主观创造的。

所以,还在犹豫要不要使用 ioGame 的开发者们,更应该讨论的是“为什么这些开发者会选择使用 ioGame”,而不是 ioGame 有没有人在使用的问题。

点击我,到语雀后台查看 ioGame 的数据


框架的安全质量

ioGame 已经接入 OSCS 墨菲安全扫描,框架的安全质量近乎100%,游戏开发者们可放心使用。更详细的可以到 ioGame 框架的安全质量、代码质量、性能 来了解。

ioGame-OSCS


愿景

让网络游戏服务器的编程变得轻松简单!

改变行业现状,降低使用难度,让游戏开发领域的生产资料公有制。

整体预览导图

image

了解 ioGame 前的准备工作

在了解 ioGame 前,这里有几个问题需要你帮忙解决,或者说你目前使用的现有框架是否能解决下面提出的问题。

下面这些问题是 ioGame 已经解决了的,并且的将来还能解决更多的问题。

如果你现在使用的框架能解决下列问题,可以不需要在花时间来学习 ioGame 了。

问题 -- 点我展开

学习成本、风险相关的问题

Q:你的使用的框架是祖传的吗?有完整的使用文档吗?如果将你当前使用的框架给到其他团队,在没有你的帮助下,该团队能顺利完成开发吗?

Q:你使用的框架是公司内部的吗?去到了新公司是否还需要重新学一套新公司的框架?或将现有公司的框架带到下个公司去使用,会给新公司带来风险吗?

Q:打算新招一个编写业务逻辑的人员,可以做到零学习成本吗?

Q:每次通信协议的变更(如 pb 内容的变更),可以做到不需要重新编译协议就能使用吗?

Q:提供游戏文档生成的辅助功能吗,支持联调文档的生成吗,支持代码即文档吗?

Q:你使用的框架会像 ioGame 这样使用语雀的每月统计,公布真实的访问数据吗?

因为成本的原因,某宝某多还没有出现能提供这种服务的商家,所以这样的统计数据也更具真实性。

Q:路由可以控制访问权限吗?如某些方法非登录玩家也能访问,有些方法只能登录后访问,有些方法即使是登录的玩家也无法访问(用于内部调用的敏感方法)。

架构相关的提问

Q:支持集群、分布式吗?是轻量级的框架吗?可以做到无需依赖任何第三方中间件或数据库就能支持集群、分布式这种真正的轻量级吗?

Q:当现有机器不能支撑现有玩家的处理时,支持机器动态增加吗?支持机器动态减少吗?动态增减会有风险吗?

Q:当有相同类型的多个逻辑服时,支持负载均衡吗?

分布式开发体验与部署相关提问

支持多服多进程的启动方式吗?

支持多服单进程的启动方式吗?

启动方式上支持随意的切换而不需要更改代码吗?

支持多服单进程的方式部署吗?

支持多服多进程多机器的方式部署吗?

部署方式上支持随意的切换而不需要更改代码吗?

业务框架相关的问题

Q:是线程安全的吗,玩家断线后,重新登录也是同个线程吗?

Q:玩家可以动态绑定指定的游戏逻辑服吗?之后的每个请求都由该游戏逻辑服来处理吗?

Q:框架支持灵活的线程扩展与设置吗?

Q:能与 spring 融合吗?除了 spring 外,还能与任何其他的框架做融合,从而使用其他框架的相关生态吗?

Q:能解决协议碎片吗?

Q:action 支持类 MVC 的编码方式吗?

action 表示一个业务动作;是接收请求,处理请求,响应数据的统称;

是路由与方法映射的一种编程模型,这种编程模型可以避免类爆炸。它提供一种清晰、灵活的方式来组织应用程序代码,将请求、响应和业务逻辑分离开来,并实现松耦合。这使得应用程序更易于维护和测试,并支持代码重用。

// 你敢相信这样就能处理长连接通信请求吗
// 这是 ioGame 编写业务逻辑时的代码,如果是新招的业务编写人员,几乎是零学习成本的。
@ActionController(1)
public class DemoAction {
    @ActionMethod(0)
    public HelloReq here(HelloReq helloReq) {
        HelloReq newHelloReq = new HelloReq();
        newHelloReq.name = helloReq.name + ", I'm here ";
        return newHelloReq;
    }
}

Q:action 请求参数除了对象外,能够支持基础类型和 List 吗?参数支持 JSR380 验证吗?

Q:action 返回值支持基础类型、List、对象的返回值吗?还是说只能支持传统的推送方式来响应数据?

Q:action 支持断言和异常机制吗?

Q:action 支持插件扩展吗?如请求信息的打印?

Q:action 支持业务代码定位与跳转吗?推送广播数据时,支持业务代码定位与跳转吗?

Q:支持类似 FlowContext 的特性吗?FlowContext 可扩展吗、可定制吗、可扩展元附加信息吗?(FlowContext 是一种对当前请求处理的上下文)

Q:能用一套业务代码,无需任何改动,同时支持多种通信协议:WebSocket、TCP 吗?

Q:能用一套业务代码,能轻松切换和扩展不同的通信协议:Protobuf、JSON 吗?

通讯方式相关的提问

支持跨进程、跨机器的单个游戏逻辑服间的相互通讯吗?

支持跨进程、跨机器与同类型多个游戏逻辑服相互通讯吗?

支持跨进程、跨机器的【游戏逻辑服】与多个【游戏对外服】通讯吗?

支持跨进程、跨机器的脉冲通讯吗?


ioGame 简介

你是否想要开发一个高性能、稳定、易用、自带负载均衡、避免类爆炸设计、可跨进程跨机器通信、集群无中心节点、集群自动化、有状态多进程的分布式的网络编程服务器呢?如果是的话,这里向你推荐一个由 java 语言编写的网络编程框架 ioGame。下面将会从多个方面来对框架做一些简单的介绍。

ioGame 是一个轻量级的网络编程框架,适用于网络游戏服务器、物联网、内部系统及各种需要长连接的场景;

ioGame 有以下特点:

  • 无锁异步化、事件驱动的架构设计
  • 同时支持 TCP、WebSocket、UDP 多种连接方式,并且可扩展
  • 支持 protobuf、json 等不同的通信协议
  • 集群无中心节点、集群自动化、分布式的设计
  • 真轻量级,不依赖任何第三方中间件或数据库就能支持集群、分布式
  • 提供多种通讯方式,且逻辑服之间可以相互跨机器通信
  • 分布式事件总线支持
  • 框架为开发者提供了同步、异步、异步回调的方法,用于逻辑服之间的相互访问
  • ioGame 是纯 javaSE 的,使得 ioGame 能与其他框架方便的进行集成、融合;如 spring ...等
  • 学习成本低,开发体验好
  • 支持多服单进程、多服多进程的启动和部署方式
  • 提供游戏文档生成的辅助功能
  • 包体小、启动快、内存占用少
  • 提供优雅的路由访问权限控制
  • 提供了灵活的线程扩展、设置
  • 具备智能的同进程亲和性
  • 具备全链路调用日志跟踪特性
  • 业务框架提供了插件机制,插件是可插拨、可扩展的
  • JSR380验证、断言 + 异常机制 = 更少的维护成本
  • action 支持自动装箱、拆箱基础类型,用于解决协议碎片的问题

ioGame 是一个专为网络编程设计的轻量级框架,它可以帮助你快速地搭建和运行自己的网络服务器。ioGame 适用于网络游戏服务器、物联网、内部系统及各种需要长连接的场景。如各种类型和规模的网络游戏,无论是 H5、手游还是 PC 游戏,无论是简单的聊天室,还是复杂的全球同服、回合制游戏、策略游戏、放置休闲游戏、即时战斗、MMORPG 等,ioGame 都可以满足你的需求。

ioGame 在打包、内存占用、启动速度等方面也是优秀的。打 jar 包后大约 15MB,应用通常会在 0.x 秒内完成启动,内存占用小。详细请看 快速从零编写服务器完整示例

在生态融合方面,ioGame 可以很方便的与 spring 集成(5 行代码);除了 spring 外,还能与任何其他的框架做融合,如:solon ...等,从而使用其他框架的相关生态。

ioGame 在架构上解决了传统框架所产生的 N*N 问题与传统架构对比)。传统架构在扩展机器时,需要借助很多第三方中间件,如:Redis、MQ、ZooKeeper ...等,才能满足整体架构的运作。通常,只要引入了需要安装的中间件才能做到扩展的,那么你的架构或者说框架,基本上与轻量级无缘了。

在轻量级方面,ioGame 不依赖任何第三方中间件或数据库就能支持集群、分布式,只需要 java 环境就可以运行。这意味着在使用上简单了,在部署上也为企业减少了部署成本、维护难度。使用 ioGame 时,只需一个依赖即可获得整个框架,而无需安装其他服务,如: Nginx、Redis、MQ、Mysql、ZooKeeper、Protobuf 协议编译工具 ...等。

ioGame 具备全链路调用日志跟踪特性,这在分布式下非常的实用。该特性为每个请求分配一个唯一标识,并记录在日志中,通过唯一标识可以快速的在日志中过滤出指定请求的信息。ioGame 提供的全链路调用日志跟踪特性更是强大,支持跨机器、跨进程。简单的说,从玩家的请求进来到结束,无论该请求经过了多少个游戏逻辑服,都能精准记录。

在通讯方式方面,大部分框架只能支持推送(广播)这一类型的通讯方式;ioGame 则提供了多种类型的通讯方式,通过对各种通讯方式的组合使用,可以简单完成以往难以完成的工作,并且这些通讯方式都支持跨进程、跨机器通信,且具备全链路调用日志跟踪。这些通讯方式分别是

  1. 请求响应(单次请求处理)
  2. 广播(推送)
  3. 单个逻辑服间的相互通讯(可跨机器通信、可跨进程通信)
  4. 与同类型多个逻辑服相互通讯(可跨多个机器通信、可跨多个进程通信)
  5. 脉冲通讯(可跨多个机器通信、可跨多个进程通信)
  6. 分布式事件总线(类似 MQ、Redis 发布订阅机制;可跨多个机器通信、可跨多个进程通信)

在编码风格上,ioGame 为开发者提供了类 MVC 的编码风格(无入侵的 Java Bean ),这种设计方式很好的避免了类爆炸。同时,框架为开发者提供了同步、异步、异步回调的方法,用于逻辑服之间的相互访问;这使得开发者所编写的代码会非常的优雅,并且具备全链路调用日志跟踪。

从 ioGame21 开始,框架添加了虚拟线程的相关支持。各逻辑服之间通信阻塞部分使用虚拟线程,这样可以很好的避免阻塞业务线程,并大幅提高了框架的吞吐量。

在线程安全方面,框架为开发者解决了单个玩家的并发问题;即使玩家重新登录后,也会使用相同的线程来消费业务,并推荐使用领域事件来解决同一房间或业务内多个玩家的并发问题。框架在线程的扩展性上提供了友好的支持,开发者可以很容易的编写出无锁并发代码,这得益于 ioGame 独有的线程执行器设计与扩展。换句话说,你不会因为并发问题烦恼。

在无锁并发方面,ioGame 提供了优雅、独特的线程执行器设计。通过该特性,开发者能轻易的编写出无锁高并发的代码。

在连接方式方面,ioGame 允许开发者使用一套业务代码同时支持多种连接方式,无需进行任何修改。ioGame 已经支持了 TCP、WebSocket 和 UDP 连接方式,并且也支持在这几种连接方式之间进行灵活切换。连接方式是可扩展的,并且扩展操作也很简单,这意味着之后如果支持了 KCP,无论你当前项目使用的是 TCP、WebSocket 还是 UDP,都可以切换成 KCP;注意了,即使切换到 KCP 的连接方式,现有的业务代码也无需改变。

在通信协议方面,ioGame 让开发者用一套业务代码,就能轻松切换和扩展不同的通信协议,如 Protobuf、JSON 等。只需一行代码,就可以从 Protobuf 切换到 JSON,无需改变业务方法。

在增减协议方面,ioGame 可以让你在新增或减少协议时,无需重启游戏对外服与 Broker(游戏网关);这样既能避免玩家断线,又能避免因新增、减少协议而重启所有机器的痛点。

在协议碎片方面,action 支持自动装箱、拆箱基础类型特性,用于解决协议碎片的问题。同时该特性除了能使你的业务代码更加清晰以外,还能大幅提高开发者在该环节的生产力。

在集群方面,ioGame 的 Broker (游戏网关)采用无中心节点、自动化的集群设计,所有节点平等且自治,不存在单点故障。集群能够自动管理和弹性扩缩,节点加入或退出时,能够自动保证负载均衡和数据一致性,不影响服务可用性。

在分布式方面,ioGame 的逻辑服使用了分布式设计**,将服务器分为游戏对外服游戏逻辑服等不同层次,并且每一层都有明确的职责和接口。这样可以提高代码可读性和可维护性,并且方便进行水平扩展

在学习成本方面,ioGame 的学习成本非常低,可以说是零学习成本,即使没有游戏编程经验,也能轻松上手。开发者只需掌握普通的 java 方法或 webMVC 相关知识,就能用框架开发业务。框架不要求开发者改变编码习惯,而是自身适应开发者的需求。

同进程亲和性方面,在同一进程内,不同 Netty 实例之间的通信,是通过内存进行传输的,不需要经过网络传输,数据传输速度极快。同进程亲和性指的是,优先访问同进程内的游戏逻辑服,当同进程内没有能处理请求的游戏逻辑服时,才会去其他进程或机器中查找能处理请求的游戏逻辑服;简单点说,框架对于请求的处理很智能,会优先将请求给同进程内的逻辑服消费。

在开发体验方面,ioGame 非常注重开发者的开发体验;框架提供了 JSR380 验证断言 + 异常机制业务代码定位,action 支持自动装箱、拆箱基础类型,用于解决协议碎片的问题 ...等。诸多丰富的功能,使得开发者的业务代码更加的清晰、简洁;

业务框架提供了插件机制,插件是可插拨、可扩展的。框架内置提供了 DebugInOutaction 调用统计业务线程监控插件各时间段调用统计插件...等插件;不同的插件提供了不同的关注点,比如我们可以使用调用、监控等插件相互配合,可以让我们在开发阶段就知道是否存在性能问题。合理利用好各个插件,可以让我们在开发阶段就能知道问题所在,提前发现问题,提前预防问题。

在分布式开发体验方面,通常在开发分布式应用时是需要启动多个进程的。这会让调试与排查问题变得非常困难,从而降低开发者的效率、增加工作量等,这也是很多框架都解决不了的问题,但 ioGame 做到了!ioGame 支持多服单进程的启动方式,这使得开发者在开发和调试分布式系统时更加简单。

与前端对接联调方面,ioGame 提供了游戏文档生成的辅助功能,可以做到代码即对接文档。简单地说,当业务代码编写完后,框架会自动生成最新的文档。如果没有游戏文档的生成,那么你将要抽出一些时间来编写、维护对接文档的工作,而且当团队人数多了之后,文档就会很乱、不同步、不是最新的、忘记更新等情况就会出现。

在部署方面,ioGame 支持多服单进程的方式部署,也支持多服多进程多机器的方式部署;在部署方式上可以随意的切换而不需要更改代码。日常中我们可以按照单体思维开发,到了生产可以选择使用多进程的方式部署。

在安全方面,所有的游戏逻辑服不需要开放端口,天然地避免了扫描攻击。由于不需要为每个逻辑服分配独立的端口,那么我们在使用诸如云服务器之类的服务时,就不需要担心端口开放权限的问题了。别小看这一个环节,通常这些小细节最浪费开发者的时间。由于我们不需要管理这些 IP:Port,这部分的工作量就自然地消失了

在模拟客户端测试方面,ioGame 提供了压测&模拟客户端请求模块。此模块是用于模拟客户端,简化模拟工作量,只需要编写对应请求与回调。除了可以模拟简单的请求外,通常还可以做一些复杂的请求编排,并支持复杂业务的压测。与单元测试不同的是,该模块可以模拟真实的网络环境,并且在模拟测试的过程中与服务器的交互是可持续的、可互动的,同时也是支持自动化的

架构灵活性方面,ioGame 的架构由三部分组成:1.游戏对外服、2.Broker(游戏网关)、3.游戏逻辑服;三者既可相互独立,又可相互融合。这意味着使用 ioGame 可以适应任何类型的游戏,因为只需通过调整部署方式,就可以满足不同类型的游戏需求。在 ioGame 中进行这些调整工作非常简单,而且不会对现有代码产生不良影响。

架构是可以动态扩缩的,游戏对外服、游戏逻辑服、Broker(游戏网关)都支持动态增加和减少。无论未来玩家数量增加或减少,我们都能够轻松应对。同时,架构是支持玩家无感知更新的,这得益于分布式设计。举例来说,如果 A 类型的游戏逻辑服需要增加一些新功能,我们可以启动 A-3、A-4 等已经支持了新功能的服务器,然后逐步将之前的 A-1 和 A-2 下线,从而实现了无感知的更新。

开发者基于 ioGame 编写的项目模块,通常是条理清晰的,得益于框架对路由的合理设计,同时也为路由提供了优雅的访问权限控制。当我们整理好这些模块后,对于其他开发者接管项目或后续的维护中,会是一个不错的帮助(模块的整理与建议)。或许现阶段你感受不到这块的威力,随着你深入地使用实践就能体会到这么设计的诸多好处与优势。

开发者基于 ioGame 编写的项目,通常是语法简洁的、高性能的、低延迟的。框架最低要求使用 JDK21,这样即可以让项目享受到分代 ZGC 带来的改进,还能享受语法上的简洁。分代 ZGC 远低于其亚毫秒级暂停时间的目标,可以在不影响游戏速度的情况下,清理掉多余的内存;这样就不会出现卡顿或者崩溃的问题了,相当于在项目中变相的引入了一位 JVM 调优大师。

综上所述,ioGame 是一个非常适合网络游戏开发的框架。可以让你轻松地创建高性能、低延迟、易扩展的游戏服务器,并且节省时间和资源。如果你想要快速地开发出令人惊艳的网络游戏,请不要犹豫,立即选择 ioGame 吧!框架屏蔽了很多复杂且重复性的工作,并可为项目中的功能模块结构、开发流程等进行清晰的组织定义,减少了后续的项目维护成本。

框架在开发、部署、压测&模拟测试 ...等,各个阶段都提供了很好的支持。相信你已经对 ioGame 有了一个初步的了解,虽然还有很多丰富的功能与特性没有介绍到,但你可以通过后续的实践过程中来深入了解。感谢你的阅读,并期待你使用 ioGame 来打造自己的游戏服务器。


ioGame 的组成

ioGame 由 [网络通信框架] 和 [业务框架] 组成

  • 网络通信框架:职责是各服务器之间的网络通信
  • 业务框架:职责是业务逻辑的处理方式和编写方式

网络通信框架 SOFABolt 是蚂蚁金融服务集团开发的一套基于 Netty 实现的网络通信框架。

  • 为了让 Java 程序员能将更多的精力放在基于网络通信的业务逻辑实现上,而不是过多的纠结于网络底层 NIO 的实现以及处理难以调试的网络问题,Netty 应运而生。
  • 为了让中间件开发者能将更多的精力放在产品功能特性实现上,而不是重复地一遍遍制造通信框架的轮子,SOFABolt 应运而生。

Bolt 名字取自迪士尼动画-闪电狗,是一个基于 Netty 最佳实践的轻量、易用、高性能、易扩展的通信框架。

业务框架 如果说 sofa-bolt 是为了让 Java 程序员能将更多的精力放在基于网络通信的业务逻辑实现上,而业务框架正是解决业务逻辑如何方便实现这一问题上。业务框架是游戏框架的一部分,职责是简化程序员的业务逻辑实现,业务框架使程序员能够快速的开始编写游戏业务。

业务框架对于每个 action (即业务的处理方法) 都是通过 asm 与 Singleton、Flyweight 、Command 等设计模式结合,对 action 的获取上通过 array 来得到,是一种近原生的方式。

单线程中,业务框架平均每秒可以执行 1152 万次业务逻辑。

业务框架性能报告JMH--点我展开

ioGameJmeter


上面是在单线程中的测试数据,业务框架平均每秒执行 1152 万次。


架构简图

ioGame

通过 ioGame 你可以很容易的搭建出一个集群无中心节点、集群自动化、分布式的网络游戏服务器!

从图中可以看出,游戏网关支持以集群方式启动多个实例。这个设计选择了集群的方式,因为游戏网关通常是无状态的,主要用于调度和转发任务

而游戏对外服、游戏逻辑服使用分布式设计,支持启动多个相同类型的服务。这意味着,当玩家数量增加时,我们可以轻松增加相应类型的游戏逻辑服以处理更多请求。

以游戏逻辑服为例,假设我们启动了两个 A 类型的游戏逻辑服,分别为 A-1 和 A-2。当玩家向 A 类型的游戏逻辑服发起多次请求时,游戏网关会使用默认的随机负载策略将请求分配给 A-1 和 A-2 来处理。

现在我们明白,游戏对外服和游戏逻辑服都支持动态增加和减少。无论未来玩家数量增加或减少,我们都能够轻松应对。架构是支持玩家无感知更新的,这得益于分布式设计。举例来说,如果 A 类型的游戏逻辑服需要增加一些新功能,我们可以启动 A-3、A-4 等已经支持了新功能的服务器,然后逐步将之前的 A-1 和 A-2 下线,从而实现了无感知的更新。

此外,框架还支持玩家动态绑定游戏逻辑服;玩家与游戏逻辑服绑定后,之后的请求都由该游戏逻辑服来处理。

除了游戏之外,ioGame 也适用于物联网相关项目。只需将图中的玩家视为具体的设备,即使存在数亿个设备,ioGame 的架构也可以轻松支持。从 2022 年开始,已经有一些物联网公司开始采用这一解决方案,并得到了很好的体验。

ioGame 适用于网络游戏服务器、物联网、内部系统及各种需要长连接的场景;

名称 扩展方式 职责
游戏对外服 分布式 与玩家连接、交互
游戏逻辑服 分布式 处理具体业务逻辑
Broker(游戏网关) 集群 调度和转发任务;

无锁异步化与事件驱动的架构设计、集群无中心节点、自带负载均衡、分布式支持、可动态增减机器、避免类爆炸的设计;

图中的每个游戏对外服、每个游戏逻辑服、每个 broker (游戏网关)都可以在单独的进程中部署,逻辑服之间可以跨进程通信(游戏对外服也是逻辑服的一种)。

游戏网关集群

broker (游戏网关)支持集群的方式部署,集群的使用是简单的,集群无中心节点、集群自动化、自带负载均衡。ioGame 本身就包含服务注册,你不需要外接一个服务注册中心,如 Eureka,ZooKeeper 等(变相的节约服务器成本)。

通过 broker (游戏网关) 的介入,之前非常复杂的负载均衡设计,如服务注册、健康度检查(后续版本提供)、到服务端的连接维护等这些问题,在 ioGame 中都不需要了,结构也简单了很多。实际上单台 broker (游戏网关) 性能已经能够满足了,因为游戏网关只做了转发。

逻辑服

逻辑服通常说的是游戏对外服和游戏逻辑服。逻辑服可以有很多个,逻辑服扩展数量的理论上限是 netty 的连接上限。

游戏对外服

对外服保持与用户(玩家)的长连接。先来个假设,假如我们的一台硬件支持我们建立用户连接的上限是 5000 人,当用户量达到 7000 人时,我们可以多加一个对外服务器来进行分流减压。由于游戏对外服扩展的简单性,意味着支持同时在线玩家可以轻松的达到百万、千万甚至更多。

即使我们启动了多个游戏对外服,开发者也不需要关心这些玩家连接到了哪个游戏对外服的问题,这些玩家总是能接收到广播(推送)消息的,因为框架已经把这些事情给做了;在玩家的角度我们只有“一个”服务器,同样的,在开发者的角度我们只有“一个”游戏对外服;

通常,有些开发者想知道游戏对外服最大支持多少玩家连接。关于这个问题,只需要搜索 Netty 相关知识即可,因为游戏对外服本质上是 Netty。

同样的,如果开发者已经熟悉了 Netty 相关知识,那么在游戏对外服的扩展上也会变得非常的容易。

在结构组合上(部署多样性)

在部署上,支持多服单进程的方式部署(类似单体应用、在分布式开发时,调试更加方便)、也支持多服多进程多机器的方式部署。

架构由三部分组成:1.游戏对外服、2.Broker(游戏网关)、3.游戏逻辑服;三者既可相互独立,又可相互融合,如:

  • 游戏对外服、Broker(游戏网关)、游戏逻辑服这三部分,在一个进程中;【单体应用;在开发分布式时,调试更加方便】
  • 游戏对外服、Broker(游戏网关)、游戏逻辑服这三部分,在多个进程中;【分布式】
  • 游戏对外服、Broker(游戏网关)这两部分在一个进程中;而游戏逻辑服在多个进程中;【类似之前游戏的传统架构】
  • 甚至可以不需要游戏对外服,只使用Broker(游戏网关)和游戏逻辑服这两部分,用于其他系统业务;

因为 ioGame 遵循面向对象的设计原则(单一职责原则、开闭原则、里式替换原则、依赖倒置原则、接口隔离原则、迪米特法则)等,所以使得架构的职责分明,可以灵活的进行组合;

游戏对外服是架构的三部分之一,默认的游戏对外服是基于 netty 实现的。如果有需要,将来我们还可以使用基于 mina、smart-socket 等通信框架编写,额外提供一个游戏对外服的实现;即使是使用 mina、smart-socket 提供的游戏对外服,也并不会影响现有的游戏逻辑服业务逻辑,因为游戏对外服满足单一职责原则,只维护用户(玩家)长连接相关的。

开发人员几乎都遇见过这么一种情况;在项目初期阶段,通常是以单体项目的方式进行开发,随着需求不断的增加与迭代,会演变成一个臃肿的项目;此时在对一个整体进行拆分是困难的,成本是极高的。甚至是不可完成的,最后导致完全的重新重构;

ioGame 提供了在结构组合上的部署多样性,通过组合的方式,在项目初期就可以避免这些拆分问题。在开发阶段中,我们可以使用单体应用开发思维,降低了开发成本。通过单体应用的开发方式,在开发分布式项目时,调试更加的方便;这既能兼顾分布式开发、项目模块的拆分,又能降低团队的开发成本;

架构其他疑问

问题一:如果一个请求经过 ioGame 再返回到客户端,对比直接用 Netty 搭个简单服务器,延迟会增加很多吗?纳秒级别还是毫秒级别?

答:是内存级。将三者部署在一起后,三者是通过内存进行传输的,不需要经过网络传输,数据传输速度极快。

原因分析,为了更好理解,现在把三者用字母代替,A.游戏对外服、B.Broker(游戏网关)、C.游戏逻辑服

  • ABC :三者在一个进程中,他们之间使用内存通信;
  • AB + C :【游戏对外服和游戏网关】在一个进程中,他们之间使用内存通信;
  • A + BC :【游戏网关和游戏逻辑服】在一个进程中,他们之间使用内存通信;

此外,ioGame 还支持同进程亲和性特性;如果我们启动了同类型的多个游戏逻辑服。【游戏网关】会优先选择与同进程的那个游戏逻辑服进行通信,使用内存通信;在同进程中没有找到时,会在同类型的多个游戏逻辑服中随机选出一个通信。

简单的一句话概括就是:同进程亲和性是指,优先访问同进程内的游戏逻辑服,当同进程内没有能处理请求的游戏逻辑服时,才会去其他进程或机器中查找能处理请求的游戏逻辑服;

所以,想要什么样的通信效果,取决于你所选择的启动方式(部署方式)。

问题二:如何更好的理解 ioGame 架构由三部分组成:1.游戏对外服、2.Broker(游戏网关)、3.游戏逻辑服;三者既可相互独立,又可相互融合?

这就好比编码时常用的三层 controller、service、dao

  • 将这三部分写在一个类也是可以的,【controller、service、dao】
  • 或者 【controller、service】+ dao
  • 又或者 controller +【service、dao】

不要将三者看成一台台生硬的机器,而是三份独立的代码。三者是可以随意组合的,这样可以满足开发者当前项目的不同需求。

ioGame 架构多样性

多服单进程、多服多进程的启动方式

架构优点

架构有很高程度的抽象,让设计者更加关注于业务,而无需考虑底层的实现、通信参数等问题。

逻辑服的位置透明性;同时,由于模块化、抽象化,使得整个架构各服务器之间耦合度很低,逻辑服注册即可用,大大增加了可伸缩性、可维护性,动态扩展变得简单而高效。由于逻辑服是注册到 Broker(游戏网关) 上的,所以逻辑服可以动态的增加、删除、改变;由于逻辑服之间耦合度较小,调试和测试的工作也是可控的;

架构比较清晰的就是,游戏对外服负责维护客户端的接入(用户、玩家的连接),游戏逻辑服专心负责业务逻辑,他们之间的调度由 Broker(游戏网关)来负责;因为架构拆分的合理,所以特别方便用 k8s 来自由伸缩部署这三种服,哪个服水位高就扩容哪个,水位过去了又可以缩容。

参考:构架简图中:对外服、游戏网关、游戏逻辑服各自的职责

通讯方式

ioGame 提供了多种类型的通讯方式,通过对各种通讯方式的组合使用,可以简单完成以往难以完成的工作,并且这些通讯方式都支持跨进程、跨机器通信的。

传统框架想要实现类似的通讯功能,只能借助大量的第三方中间件,而 ioGame 则无需任何中间件就能实现这些通讯功能。这意味着在使用上简单了,在部署上也为企业减少了部署成本、维护难度。

这些通讯方式都是通过扩展实现的,重点是支持可扩展。

一般传统的框架只提供了接收请求,当请求处理完后使用推送的方式将数据响应给请求端。但在使用 ioGame 时,不要被过去的传统框架束缚住,可以做任何大胆的设计,因为通讯方式足够丰富;

img

总的来说,ioGame 支持的通讯方式分为两大类,分别是路由类和主题类;下面分别对这些通讯方式做一些应用场景举例介绍。

框架对这些通讯方式提供了代码调用点的日志,简单点说就是框架可以让开发者知道,是在哪一行代码中触发的业务逻辑。

我们可以想象一下,假如框架没有提供代码调用点的日志会是什么样的;比如,游戏前端发送一个业务请求到游戏服务器中,但是处理这个请求的业务方法,会触发多个响应(通常是推送、广播)给游戏前端。一但时间久了,开发者是很难知道分别响应了哪些业务数据给游戏前端,特别是一些二手项目;所以这将是一个灾难性的问题,因为这会耗费大量的时间来寻找这些相关的业务代码。

多类型的通讯方式的详细内容--点我展开

1.单次请求处理

1.1 请求、无响应

action 处理逻辑时,使用是就是这种通讯方式。将方法声明为 void,无需给请求端任何响应;本质上是不给请求端推送任何数据;

当请求端发起请求后,游戏逻辑服不会发送任何响应给请求端。可以用在在网络通讯中,存在着不需要接收方回执确认的调用模型,如数据采集的场景: 打点采集、日志传输、metrics上报等。

在写 action 时,将方法返回值声名为 void 就表示处理 请求、无响应的。

1.2 请求、响应

action 处理逻辑时,使用是就是这种通讯方式。将数据 return 给请求端;本质上是将数据推送到请求端;

请求、响应是在游戏开发中常见的通讯模式,也就是通讯的一方发出请求,而远程通讯的对方做出响应,也就是常说的请求/响应模式。

比如:装备的升级、人物的升级、玩家的移动、抽奖、游戏前端到某一个场景时需要从游戏服务端获取一些对应的场景配置等;

在写 action 时,方法有返回值的就表示处理 请求、响应的,框架会将这个返回值给到请求端。

2.推送

2.1 指定单个或多个用户广播

服务器主动的将数据给【单个或多个玩家】

向一个或多个指定的用户(玩家)主动发送一些数据。比如:

  • 给指定的在线玩家发送一些奖励。
  • 给在同一个房间内的玩家广播一些数据,如某一个玩家射击子弹,把这子弹的数据广播给房间内的其他玩家。如几个玩家在同一个房间内打牌,某个玩家出牌后,把这张牌的数据广播给房间内的其他玩家。

2.2 全服广播

服务器主动的将数据给【全服在线的玩家】

给全服的所有在线玩家广播消息,如广播公告、即将停服维护等。

详细示例可参考:广播示例

3.单个逻辑服间的相互通讯

3.1【游戏逻辑服】与单个【游戏逻辑服】通信;- 有返回值(可跨进程)

1 : 1 的请求,也就是【单次请求处理】的通讯方式;支持进程内、跨进程、跨机器;开发者无需关心,一切都是自动的。

比如:我们有两个游戏逻辑服,分别是:a.天气预报逻辑服、b.战斗逻辑服。

现在我们设想一个回合制游戏的战斗场景,需要配合天气,根据天气来增强或者减弱某个英雄的能力。

那么在战斗开始前,战斗逻辑服只需要向游戏网关发起一个获取当前天气的请求,就可以得到当前的天气信息了,在根据当前的天气数据来增强或减弱该英雄的能力。

又比如:a.大厅逻辑服、b.奖励发放逻辑服。大厅记录着一些数据(房间总数),奖励发放逻辑服根据当前的房间数量,来生成不同奖品,随机发放给在线用户。

详细示例可参考:逻辑服与逻辑服之间的交互示例

3.2【游戏逻辑服】与单个【游戏逻辑服】通讯; - 无返回值(可跨进程)

1 : 1 的请求,也就是【单次请求处理】的通讯方式;支持进程内、跨进程、跨机器;开发者无需关心,一切都是自动的。

比如:我们有两个游戏逻辑服,分别是:a.匹配逻辑服、b.房间逻辑服。

业务场景如下,多个玩家在开始游戏前需要匹配。这里假设有两个玩家,当匹配完成后,给这两个玩家返回所匹配到的房间信息。

具体实现如下,两个玩家分别向匹配逻辑服发送匹配请求,匹配逻辑服收到玩家的请求后进行逻辑处理,并成功的把这两个玩家匹配到一起,此时我们把两个匹配到一起的玩家先称为匹配结果。

匹配逻辑服只负责匹配相关的算法逻辑,所以在匹配逻辑服中,我们可以把匹配结果给到房间逻辑服,因为与匹配相关的工作已经完成了。

在匹配逻辑服中,我们可以向房间逻辑服发起一个(单个逻辑服与单个逻辑服通信请求 - 无返回值)的请求,当房间逻辑服拿到匹配结果,根据匹配结果来创建房间。房间创建完成后把结果用推送(广播)给这两名玩家。

为什么要用无返回值的通信请求呢,因为匹配逻辑服并不关心房间的创建。

详细说明可参考:逻辑服与逻辑服之间的交互-无返回值

4.与同类型多个逻辑服相互通讯

4.1【游戏逻辑服】与同类型多个【游戏逻辑服】通信;- 可跨进程

1 : N 的请求,本质上是【单次请求处理】;支持进程内、跨进程、跨机器;开发者无需关心,一切都是自动的。

从多个游戏逻辑服中得到结果;

如: 【象棋逻辑服】有 3 台,分别是:《象棋逻辑服-1》、《象棋逻辑服-2》、《象棋逻辑服-3》,这些逻辑服可以在不同的进程中

我们可以在大厅逻辑服中向【同类型】的多个游戏逻辑服请求,意思是大厅发起一个向这 3 台象棋逻辑服的请求,框架会收集这 3 个结果集(假设结果是:当前服务器房间数)。

当大厅得到这个结果集,可以统计房间的总数,又或者说根据这些信息做一些其他的业务逻辑;这里只是举个例子。实际当中可以发挥大伙的想象力。

详细示例可参考:请求同类型多个逻辑服通信结果

其中配合动态绑定逻辑服节点;可以实现LOL、王者荣耀匹配后动态分配房间

4.2【游戏逻辑服】与多个【游戏对外服】通信;- 可跨进程

从多个游戏对外服中得到结果;

可以向游戏对外服拿一些玩家数据,或者是其他的一些操作。框架在游戏对外服中提供了 ExternalBizRegion 接口,可以使得开发者在游戏对外服中的扩展变得很简单。

框架利用这一通讯特性与 ExternalBizRegion 扩展,在不到 15 行的有效代码中,就实现了,如:查询用户(玩家)是否在线、强制用户(玩家)下线....等功能,从而实现了登录功能的增强:重复登录、顶号这些业务。

具体扩展与使用可以参考 获取游戏对外服的数据与扩展 文档

5.脉冲通讯

5.1 脉冲通讯

脉冲通讯与发布订阅类似,但是它除了具备发布订阅的无需反馈的方式,还增加了接收消息响应的动作,这是它与发布订阅的重要区别。

需要注意的是,脉冲通讯只是一种通讯方式,不能完全取代发布订阅,而是适用于一些特殊的业务场景。虽然在理论上,这些特殊的业务场景可以使用发布订阅来完成,但这会让代码变得复杂。

脉冲通讯方式 - 文档


6.分布式事件总线

6.1 分布式事件总线

分布式事件总线是 ioGame 提供的通讯方式之一。该通讯方式与 Guava EventBus、Redis 发布订阅、MQ ... 等产品类似。

如果使用 Redis、MQ ...等中间件,需要开发者额外的安装这些中间件,并支付所占用机器的费用;使用 Guava EventBus 则只能在当前进程中通信,无法实现跨进程。

而 ioGame 提供的分布式事件总线,拥有上述两者的优点。此外,还可以有效的帮助企业节省云上 Redis、 MQ 这部分的支出。

事件发布后,除了当前进程所有的订阅者能接收到,远程的订阅者也能接收到(支持跨机器、跨进程、跨不同类型的多个逻辑服)。可以代替 redis pub sub 、 MQ ,并且具备全链路调用日志跟踪,这点是中间件产品做不到的。

ioGame 分布式事件总线,特点

  • 使用方式与 Guava EventBus 类似
  • 具备全链路调用日志跟踪。(这点是中间件产品做不到的)
  • 支持跨多个机器、多个进程通信
  • 支持与多种不同类型的多个逻辑服通信
  • 纯 javaSE,不依赖其他服务,耦合性低。(不需要安装任何中间件)
  • 事件源和事件监听器之间通过事件进行通信,从而实现了模块之间的解耦
  • 当没有任何远程订阅者时,将不会触发网络请求。(这点是中间件产品做不到的)

分布式事件总线 - 文档


最后,发挥你的想象力,把这些类通讯方式用活,可以满足很多业务。

快速入门

业务交互

img

抽象的说,游戏前端与游戏服务器的的交互由上图组成。游戏前端与游戏服务器可以自由地双向交互,即发送和接收业务数据。业务数据由 .proto 文件作为载体,在前端和后端之间进行编码和解码。.proto 文件是对业务数据的描述载体,定义了数据类型和消息类型,以及它们的属性和规则。

通过这种方式,游戏前端和游戏服务端可以建立连接,并开始相互传递业务数据,处理各自的业务。以上是对游戏前端与游戏服务器之间交互方式的介绍。接下来,我们将编写一个简单的游戏业务处理示例,并定制一个适合我们需求的业务数据协议。

协议文件

协议文件是对业务数据的描述载体,用于游戏前端与游戏服务器的数据交互。Protocol Buffers 是 Google 公司开发的一种数据描述语言,也简称 PB。协议文件描述还可以是 json、xml或者任意自定义的,因为最后传输时会转换为二进制,但游戏开发中 PB 是目前的最佳。

游戏前端

游戏前端的展现可以是 UnityUECocos CreatorFXGLGodot 或者其他的游戏引擎。这些游戏引擎只是展现游戏画面的一种形式,数据交互则由通信来完成(TCP、UDP 等)。


快速入门代码示例

这里主要介绍游戏服务器相关的,下面这个示例介绍了服务器编程可以变得如此简单。


协议文件定义

首先我们自定义一个协议文件,这个协议文件作为我们的业务载体描述。这个协议是纯 java 代码编写的,使用的是 jprotobuf,jprotobuf 是对 google protobuf 的简化使用,性能同等。

可以把这理解成 DTO、POJO 业务数据载体等,其主要目的是用于业务数据的传输;

/** 请求 */
@ProtobufClass
@FieldDefaults(level = AccessLevel.PUBLIC)
public class HelloReq {
    String name;
}

Action

游戏服务器的编程,游戏服务器接收业务数据后,对业务数据进行处理;下面这段代码可以同时支持 TCP、WebSocket、UDP 通信方式。

示例代码中展示了玩家的请求与响应处理,还展示了跨服(跨进程、跨机器)的请求处理的示例。

@ActionController(1)
public class DemoAction {
    @ActionMethod(0)
    public HelloReq here(HelloReq helloReq) {
        // 业务数据
        var newHelloReq = new HelloReq();
        newHelloReq.name = helloReq.name + ", I'm here ";
        return newHelloReq;
    }

    // 注意,这个方法只是为了演示而写的;(ioGame21 开始支持)
    // 效果与上面的方法一样,只不过是用广播(推送)的方式将数据返回给请求方
    @ActionMethod(0)
    public void here(HelloReq helloReq, FlowContext flowContext) {
        // 业务数据
        var newHelloReq = new HelloReq();
        newHelloReq.name = helloReq.name + ", I'm here ";

        flowContext.broadcastMe(newHelloReq);
    }

    // 跨服调用示例,下面分别展示了同步与异步回调的写法
    void testShowInvokeModule(FlowContext flowContext) {
        /*
         * 框架为跨服请求提供了同步、异步、异步回调的编码风格 api。(ioGame21 开始支持)
         */
        var cmdInfo = CmdInfo.of(1,0);
        var yourData = ... 你的请求参数
        
        // 跨服请求(异步回调 - 无阻塞)-- 路由、请求参数、回调。
        flowContext.invokeModuleMessageAsync(cmdInfo, yourData, responseMessage -> {
            var helloReq = responseMessage.getData(HelloReq.class);
             // --- 此异步回调,具备全链路调用日志跟踪 ---
            log.info("异步回调 : {}", helloReq);
        });


        // 跨服请求(同步 - 阻塞)-- 路由、请求参数。
        ResponseMessage responseMessage = flowContext.invokeModuleMessage(cmdInfo, yourData);
        var helloReq = responseMessage.getData(HelloReq.class);
        log.info("同步调用 : {}", helloReq);
    }
}

一个方法(here)在业务框架中表示一个 Action(一个业务动作)。

方法声明的参数是用于接收前端传入的业务数据,在方法 return 时,数据就可以被游戏前端接收到。程序员可以不需要关心业务框架的内部细节。

从上面的示例可以看出,这和普通的 java 类并无区别,同时这种设计方式避免了类爆炸。如果只负责编写游戏业务,那么对于业务框架的学习可以到此为止了。

游戏编程就是如此简单!

问:我可以开始游戏服务器的编程了吗?

是的,你已经可以开始游戏服务器的编程了。

访问示例(控制台)

当我们访问 here 方法时(通常由游戏前端来请求),控制台将会打印

┏━━━━━ Debug. [(DemoAction.java:4).hello] ━━━━━ [cmd:1-0 65536] ━━━━━ [逻辑服 [xxx逻辑服] - id:[76526c134cc88232379167be83e4ddfc]]
┣ userId: 1
┣ 参数: active : HelloReq(id=101, name=塔姆)
┣ 响应: HelloReq(name=塔姆, I'm here )
┣ 时间: 1 ms (业务方法总耗时)
┗━━━━━ [ioGameVersion] ━ [线程:User-8-2] ━ [连接方式:WebSocket] ━ [traceId:956230991452569600] ━

控制台打印说明

Debug. [(DemoAction.java:4).here]:
       表示执行业务的是 DemoAction 类下的 here 方法,4 表示业务方法所在的代码行数。
       在工具中点击控制台的 DemoAction.java:4 这条信息,就可以跳转到对应的代码中(快速导航到对应的代码)。

userId : 当前发起请求的 用户 id。

参数 : 通常是游戏前端传入的值。

响应 : 通常是业务方法返回的值 ,业务框架会把这个返回值推送到游戏前端。

时间 : 执行业务方法总耗时,我们可根据业务方法总耗时的时长来优化业务。

路由信息 : 路由 是唯一的访问地址。

ioGameVersion:表示当前所使用的 ioGame 版本。

线程:当前执行 action 所使用的线程。

traceId:全链路调用日志跟踪 id,每个请求唯一。

逻辑服:当前游戏逻辑服与其 id

连接方式:当前玩家所使用的连接方式,TCP、WebSocket、UDP

有了以上信息,游戏开发者可以很快的定位问题。如果没有可视化的信息,开发中会浪费很多时间在前后端的沟通上。问题包括:

  • 是否传参问题 (游戏前端说传了)
  • 是否响应问题(游戏后端说返回了)
  • 业务执行时长问题 (游戏前端说没收到响应, 游戏后端说早就响应了)

其中代码导航可以让开发者快速的跳转到业务类对应代码中,在多人合作的项目中,可以快速的知道业务经过了哪些方法的执行,使得我们可以快速的进行阅读或修改;


适合人群?

  1. 长期从事 web 内部系统开发人员, 想了解游戏的
  2. 刚从事游戏开发的
  3. 未从事过游戏开发,但却对其感兴趣的
  4. 对设计模式在实践中的应用和 sofa-bolt 有兴趣的学习者
  5. 可以接受新鲜事物的
  6. 想放弃祖传代码的

推荐实际编程经验一年以上的人员


更多示例代码(广播、跨服)

在上面的简单示例中,我们展示了请求/响应、跨服调用时的同步与异步回调的写法。这里,将展示更多使用通讯相关的方法,分别是:

  1. 广播(推送)

    1. 全服广播
    2. 指定单个用户广播
    3. 指定多个用户广播
  2. 跨服调用(单个游戏逻辑服之间的交互)

    1. 同步的调用方式
    2. 异步回调的调用方式
  3. 跨服调用(请求同类型多个游戏逻辑服通信结果)

    1. 同步的调用方式
    2. 异步回调的调用方式

除了上面这三种的通讯方式外,还有更多通讯方式,可阅读相关文档:请先读我-通讯相关

@ActionController(1)
public class TestAction {
    ... ...省略部分代码
    // ======== 广播相关 ========
    // https://www.yuque.com/iohao/game/qv4qfo
    // 特点:可向任意玩家主动发送消息
    public void broadcast() {
        // ======== 广播相关 --- 全服、指定用户(单个、多个) ========
        // 全服广播 - 路由、业务数据
        flowContext.broadcast(cmdInfo, yourData);

        // 广播消息给单个用户 - 路由、业务数据、userId
        long userId = 100;
        flowContext.broadcast(cmdInfo, yourData, userId);

        // 广播消息给指定用户列表 - 路由、业务数据、userIdList
        List<Long> userIdList = new ArrayList<>();
        userIdList.add(100L);
        userIdList.add(200L);
        flowContext.broadcast(cmdInfo, yourData, userIdList);

        // ======== 广播相关 --- 给自己广播 ========
        // 给自己发送消息 - 路由、业务数据
        flowContext.broadcastMe(cmdInfo, yourData);

        // 给自己发送消息 - 业务数据
        // 路由则使用当前 action 的路由。
        flowContext.broadcastMe(yourData);
    }

    // ======== 单个游戏逻辑服之间的交互 - 跨服调用 ========
    // https://www.yuque.com/iohao/game/anguu6
    // 特点:可接收响应
    void invokeModuleMessage() {
        // ======== 单个游戏逻辑服之间的交互 --- 同步 ========
        // 路由
        ResponseMessage responseMessage = flowContext.invokeModuleMessage(cmdInfo);
        RoomNumMsg roomNumMsg = responseMessage.getData(RoomNumMsg.class);
        log.info("同步调用 : {}", roomNumMsg.roomCount);

        // 路由、请求参数
        ResponseMessage responseMessage2 = flowContext.invokeModuleMessage(cmdInfo, yourData);
        RoomNumMsg roomNumMsg2 = responseMessage2.getData(RoomNumMsg.class);
        log.info("同步调用 : {}", roomNumMsg2.roomCount);

        // ======== 单个游戏逻辑服之间的交互 --- 异步回调 ========
        // --- 此回调写法,具备全链路调用日志跟踪 ---
        // 路由、回调
        flowContext.invokeModuleMessageAsync(cmdInfo, responseMessage -> {
            RoomNumMsg roomNumMsg = responseMessage.getData(RoomNumMsg.class);
            log.info("异步回调 : {}", roomNumMsg.roomCount);
        });

        // 路由、请求参数、回调
        flowContext.invokeModuleMessageAsync(cmdInfo, yourData, responseMessage -> {
            RoomNumMsg roomNumMsg = responseMessage.getData(RoomNumMsg.class);
            log.info("异步回调 : {}", roomNumMsg.roomCount);
        });
    }

    // ======== 单个游戏逻辑服之间的交互 - 跨服调用 ========
    // https://www.yuque.com/iohao/game/anguu6
    // 特点:异步,不需要接收响应
    void invokeModuleVoidMessage() {
        // 适合不需要接收响应的业务,默认异步
        
        // 路由
        flowContext.invokeModuleVoidMessage(cmdInfo);
        // 路由、请求参数
        flowContext.invokeModuleVoidMessage(cmdInfo, yourData);
    }

    // ======== 请求同类型多个逻辑服通信结果 - 跨同类型多个游戏逻辑服调用 ========
    // https://www.yuque.com/iohao/game/rf9rb9
    // 特点:可同时接收多个游戏逻辑服的响应
    void invokeModuleCollectMessage() {
        // ======== 请求同类型多个逻辑服通信结果 --- 同步 ========
        // 路由
        ResponseCollectMessage response = flowContext.invokeModuleCollectMessage(cmdInfo);

        // 打印其他游戏逻辑服所响应的数据
        for (ResponseCollectItemMessage message : response.getMessageList()) {
            RoomNumMsg roomNumMsg = message.getData(RoomNumMsg.class);
            log.info("同步调用 : {}", roomNumMsg.roomCount);
        }

        // 路由、请求参数
        ResponseCollectMessage response2 = flowContext.invokeModuleCollectMessage(cmdInfo, yourData);
        log.info("同步调用 : {}", response2.getMessageList());

        // ======== 请求同类型多个逻辑服通信结果 --- 异步回调 ========
        // --- 此回调写法,具备全链路调用日志跟踪 ---

        // 路由、回调
        flowContext.invokeModuleCollectMessageAsync(cmdInfo, responseCollectMessage -> {
            List<ResponseCollectItemMessage> messageList = responseCollectMessage.getMessageList();

            for (ResponseCollectItemMessage message : messageList) {
                RoomNumMsg roomNumMsg = message.getData(RoomNumMsg.class);
                log.info("异步回调 : {}", roomNumMsg.roomCount);
            }
        });

        // 路由、请求参数、回调
        flowContext.invokeModuleCollectMessageAsync(cmdInfo, yourData, responseCollectMessage -> {
            log.info("异步回调 : {}", responseCollectMessage.getMessageList());
        });
    }
}

@ToString
@ProtobufClass
public class RoomNumMsg {
    public int roomCount;
}

框架内置功能

内置多种可选模块,可按需选择,以方便应用开发:

游戏服务器框架内置功能详细--点我展开
  • 领域事件 (轻量级单机最快MQ -- disruptor;可为你的系统实现类似 Guava-EventBus、Spring 事件驱动模型 ApplicationEvent、业务解耦、规避并发、不阻塞主线程... 等,各种浪操作。)
  • 任务延时器 (将来某个时间可对任务进行执行、暂停、取消等操作,并不是类似 Quartz 的任务调度)
  • 多环境切换 (不同运行环境下的配置支持)
  • light-jprotobuf (补足 jprotobuf 不能让多个对象在单个 .proto 源文件中生成的需求,并简化jprotobuf对源文件的注释)
  • 分布式锁 (基于Redisson的简单实现)
  • 压测&模拟客户端请求
  • room 桌游、房间类,该模块是桌游类、房间类游戏的解决方案。比较适合桌游类、房间类的游戏基础搭建,基于该模型可以做一些如,炉石传说、三国杀、斗地主、麻将 ...等类似的桌游。或者说只要是房间类的游戏,该模型都适用。比如,CS、泡泡堂、飞行棋、坦克大战 ...等。

内置的其他功能:


快速从零编写服务器完整示例

如果觉得 ioGame 适合你,可以看一下 快速从零编写服务器完整示例 。在这个示例中,你可以用很少的代码实现一个完整的、可运行的、高性能的、稳定的服务器。


参考

什么是 Action

快速从零编写服务器完整示例

广播(推送)相关示例与文档

逻辑服与逻辑服之间的交互示例

构架简图中:对外服、游戏网关、游戏逻辑服各自的职责

安装与使用ioGame

参考 https://www.yuque.com/iohao/game/wsgmba

iogame's People

Contributors

iohao avatar kyochou avatar licheng1013 avatar oldshen avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

iogame's Issues

业务参数自动装箱和拆箱,基础类型:int、long

在实际开发中,会有一些基础类型的业务参数,比如需要经常用到下面类似的

通过 装备-衣服-id 得到衣服装备详细
通过 装备-头盔-id 得到头盔装备详细
.... 
通过 xx-id 得到xx数据 等

这些参数基本都是相同的,如果每一个类似这样的请求都创建一个数据协议,会产生比较多的碎片协议,如

// 装备-头盔-id pb
@ProtobufClass
@FieldDefaults(level = AccessLevel.PUBLIC)
public class EquipHelmetIdPb {
    int id;
}

// 装备-衣服-id pb
@ProtobufClass
@FieldDefaults(level = AccessLevel.PUBLIC)
public class EquipClothesIdPb {
    int id;
}

// 业务action
@ActionController(1)
public class DemoAction {
    @ActionMethod(0)
    public xxx getEquipPb(EquipClothesIdPb equipClothesIdPb) {
    	// 通过装备id , 查询装备
    	int equipId = equipClothesIdPb.id;
        return ...
    }
}

业务框架提供的内置支持

对于这种情况,业务框架将提供一些较好的支持。
首先,业务框架会提供一个通用的 pb 类型,如下

@ProtobufClass
@FieldDefaults(level = AccessLevel.PUBLIC)
public class IntPb {
    int intValue;
}

在代码中的使用,如下伪代码

@ActionController(1)
public class DemoAction {
    @ActionMethod(0)
    public xxx getEquipPb(IntPb equipIdPb) {
    	// 通过装备id , 查询装备
    	int equipId = equipIdPb.intValue;
        return ...
    }
}

由于是业务参数基础包装类型-int,实际上我们还可以这样使用,下面这种用法与上面是等价的。
这种实际上就是有点像是 java 基础类型的自动装箱和拆箱

@ActionController(1)
public class DemoAction {
    @ActionMethod(0)
    public xxx getEquipPb(int equipId) {
    	
        return ...
    }
}

关于对接文档的生成,即使你写成 (int equipId) 这种参数形式,框架会生成大概这样的文档


==================== DemoAction 装备相关 ====================
路由: 1 - 0  --- 【查询装备衣服】 --- 【DemoAction:44】【getEquipPb】
    方法参数: com.iohao.game.common.msg.IntPb
    方法返回值: com.xxx.YourEquipPb # 就是你项目当中的装备pb


提供 bolt UserProcessor 用户线程池设置策略

分离IO线程池与用户线程池,这样服务器可以在同一时间内处理更多的请求。

框架提供 UserProcessorExecutorStrategy 接口。

主要用于给 UserProcessor 构建 Executor 的策略;

框架会在启动时,如果检测到 UserProcessor 实现了 UserProcessorExecutorAware 接口,就会触发一次

通过该接口,开发者可以给 UserProcessor 配置 Executor;

开发者可以根据自身业务来做定制
see IoGameGlobalConfig.userProcessorExecutorStrategy

框架源码、线程、容灾等问题

框架源码的交互时序方面:

  1. 客户端与external建立连接的入口在哪?(accept方法位置,想看看整体流程)
  2. broker的路由寻址逻辑和负载均衡具体在哪里实现的(从一个请求发送过来,到找到指定的逻辑服务器这块逻辑)
  3. 网关和逻辑服之前的通信,还有连接管理的具体位置在哪?源代码是哪个class里面写的能否指一下类名

框架的线程池方面:

  1. 对外、网关、逻辑服几种线程(例如:IO线程 和 工作线程 )具体如何划分的。
  2. 网络模型是什么样子(nio,bio,aio还是io多路复用),顺便能指明一下class类就更好了。

状态持久化针对容灾
1.内部session的状态存储目前都存储在内存,如果服务部署在云端,出现容器漂移、逐出是很常见的事情(物理机也会出现死机等问题),这种情况用户连接都会断开,(此时这种有状态的数据---包含但不限于用户session,是否可以有扩展的口子让使用者自己实现持久化逻辑) 使用者自己就可以重写实现这些有状态数据的持久化细节。(宕机后客户端重连仍然可以恢复用户的有状态数据)
2. 广播消息推送的时候,广播是临时存储在broker内存里面的吗?如果需要广播发送给uid=1,2,3,4,5这5个用户,但是由于中间环节出现不可预知的问题(例如某一台broker服务宕机了),广播是否就没了。如果涉及到一些财务相关的业务,使用广播给2个正在进行交易的用户发送广播,但是其中一个用户external连接出现了问题,客户端有一些重连的逻辑,重连上之后,希望能有一个重试机制这样的。

疑问点:

  1. broker的实现,为什么选择自己实现,为何不采用相对比较成熟的方案,例如zk,etcd。

是否提供热更新的能力?

在游戏里面,不停服更新是很重要的一个能力,在更新的时候玩家无感,会很大程度的提升用户体验,如果有希望提供个完整的例子,谢谢~

标准 action 规则

标准 action 映射规则

  1. 业务方法上添加注解 ActionMethod
  2. 业务方法的访问权限必须是:public
  3. 业务方法不能是:static
  4. 业务方法需要是在 action 类中声明的方法

简单的说,标准的 action 是非静态的,且访问权限为 public 的方法。
术语说明:在 action 类中提供的业务方法通常称为 action

ActionAfter 异常信息携带到请求端

see https://gitee.com/iohao/iogame/issues/I5KZTP

业务框架的异常机制,只保存错误码,不会保存错误码对应的异常信息,这是因为:

从字段精简的角度,我们不可能每次响应都带上完整的异常信息给客户端排查问题,因此,我们会定义一些响应码,通过编号进行网络传输,方便客户端定位问题。

但有一种情况是会携带异常信息,就是开启 JSR380 时,具体参考 业务框架开启JSR验证


配合游戏文档生成
https://www.yuque.com/iohao/game/irth38

==================== TankAction 坦克相关 ====================
路由: 2 - 2  --- 【玩家进入房间】 --- 【TankAction:135】【enterRoom】
    方法参数: com.iohao.game.collect.proto.tank.TankEnterRoom
    方法返回值: com.iohao.game.collect.proto.tank.TankEnterRoom
 
 
路由: 2 - 5  --- 【坦克移动】 --- 【TankAction:101】【tankMove】
    方法参数: com.iohao.game.collect.proto.tank.TankLocation
    广播推送: com.iohao.game.collect.proto.tank.TankLocation
 
路由: 2 - 6  --- 【坦克射击(发射子弹)】 --- 【TankAction:74】【shooting】
    触发异常: (方法有可能会触发异常)
    方法参数: com.iohao.game.collect.proto.tank.TankBullet
    广播推送: com.iohao.game.collect.proto.tank.TankBullet
 
==================== 其它广播推送 ====================
路由: 2 - 11  --- 广播推送: com.iohao.game.collect.proto.common.UserInfo

==================== 错误码 ====================
 -1002 : 路由错误码,一般是客户端请求了不存在的路由引起的 
 -1001 : 参数验错误码 
 -1000 : 系统其它错误 
 -1 : class 不存在 
 201 : 子弹不存在或不足 

在生成的游戏文档中,可以看见所有的错误码;
如果做国际化,放在前端会比较好做一些。

Request for the English documentation

Hi @iohao,
I appreciate your effort to create and share the awesome repo. However, since it is hard to approach the project in Chinese, I would be grateful if you could spare some time to provide the English version for documentation.
Thank you so much.

Best regards,
Kong

游戏对外服 netty 业务编排钩子接口

新增 ChannelPipelineHook netty 业务编排的处理器钩子接口,用于游戏对外服。

ExternalServerBuilder 新增 channelPipelineHook ,用于自定义 netty 业务编排的处理器。

废弃标记 ExternalServerBuilder.channelHandlerProcessors 属性及其相关地方为过期,由 ChannelPipelineHook 来代替,使用示例如下。

使用示例代码

public class ExternalServerBuilderApplication {
    public static void main(String[] args) {
        // netty ChannelHandler 编排
        var myChannelPipelineHook = new ChannelPipelineHook() {
            @Override
            public void initChannelPipeline(ChannelPipeline pipeline) {
                pipeline.addLast("ExternalBizHandler", new ExternalBizHandler());

                // pipeline.addFirst("ssl", ssl)
            }
        };

        // 游戏对外服构建器
        ExternalServerBuilder builder = ExternalServer.newBuilder(10100);
        builder.channelPipelineHook(myChannelPipelineHook);
    }
}

通常情况下,这样的编排方式也会更加清晰,因为代码统一在这里做编排了。

开发者在自定义业务编排时,可以通过这个钩子接口,比如添加一个 SSL。

注意事项:在调用 hook 前,会经过 ExternalChannelInitializerCallback.initChannelPipeline(SocketChannel) 。
ExternalChannelInitializerCallback 接口的实现类有

  1. ExternalChannelInitializerCallbackWebsocket
  2. ExternalChannelInitializerCallbackTcp

这些实现类中,会给 ChannelPipeline 添加上一些默认的处理器,通常是编解码相关的。

逻辑服请求网关超时

当我瞬间发送几十条的时候,就会出现部分超时现象。。网关那打了日志,发现网关实际收到的频率偏低。感觉像是有大量的请求过了一会才抵达网关(网关会二次转发给对外服,俩者之间目前没产生超时现象)。。。不知道是怎么回事,希望大神请指点一下,我们现在打算部署到生产,结果测试的时候出现了这个现象

逻辑服和网关断连

您好,如果逻辑服频繁请求网关,会出现短时间的断连现象,网关那边会显示此逻辑服下线。过一会会重新连接。

业务框架与网络通信框架解耦

新增 ChannelContext 通信通道接口
用于对 bolt AsyncContext、netty Channel 的包装,这样可以使得业务框架与网络通信框架解耦。
为将来 ioGame 实现绳量级架构的使用做准备。

简化元附加信息的使用

简化元附加信息的使用

在处理 action 时,我们可以通过 FlowContext.userId 可以很方便得到当前用户(玩家)id。

如果开发者想在处理 action 时,携带上一些自定义的信息时,可以通过元附加信息特性来完成。

比如保存当前玩家的英雄角色 id,或者玩家的昵称,又或者是你的项目的 userId 是 string 或其他类型则可以通过元信息这一特性来兼容。

简单的说,就是你想在 FlowContext 中获取一些用户(玩家)特有的信息数据时,可以通过这个特性来实现。

新增 元信息接口 Attachment

/**
 * 元信息接口
 * <pre>
 *     注意:
 *     框架默认使用的是 protobuf 编解码,所以建议子类添加 ProtobufClass 注解
 * </pre>
 *
 * @author 渔民小镇
 * @date 2022-12-11
 */
public interface Attachment extends Serializable {
    /**
     * get userId
     *
     * @return userId
     */
    long getUserId();
}

支持原生protobuf

jprotobuf和原生protobuf序列化结果不一致, jprotobuf存在默认值和null, 会导致序列化完长度与客户端不同. 能否支持原生protobuf

集群网关下一直报错

飞书20220823-170715
您好,我使用集群模式之后,虽然可以使用,但是一直报错,这个报错端口号也会一直变。

严格登录、路由权限控制

严格登录指的是,如果玩家没有登录,是不能访问其他业务方法的(即 action)。

var accessAuthenticationHook = ExternalGlobalConfig.accessAuthenticationHook;
// 表示登录才能访问业务方法
accessAuthenticationHook.setVerifyIdentity(true);
// 添加不需要登录(身份验证)也能访问的业务方法 (action)
accessAuthenticationHook.addIgnoreAuthenticationCmd(1, 1);

通常这段代码放到游戏对外服中,因为是在游戏对外服做的权限控制。

当 setVerifyIdentity = true 时,是不能访问任何业务方法的,包括开发者编写的登录业务方法。但我们可以放开权限,使得玩家可以访问我们的业务方法。上面的伪代码中,放开了路由 1-1 的访问权限,这样不需要登录也可以访问游戏逻辑服的业务方法了。

默认配置下,setVerifyIdentity = false ,就是不登录就可以访问所有的业务方法。

文档
https://www.yuque.com/iohao/game/tywkqv#qEvtB

参数验证问题

您好,不知道现在是否支持例如springboot那种参数验证分组功能。同一个实体类,我可以在不同方法上验证不同的参数

支持多种协议:protobuf、json,并支持可扩展

ioGame 支持同样的一套业务代码,无需做变更,可以同时支持多种协议:protobuf、json,并可扩展。在 ioGame 中切换协议是简单的,只需要一行代码。

使用文档
https://www.yuque.com/iohao/game/uq2zrltrc7to27bt

public class JsonApplication {
    public static void main(String[] args) {
        // 设置 json 编解码。如果不做设置,默认使用 jprotobuf
        IoGameGlobalSetting.me().setDataCodec(new JsonDataCodec());

        // 游戏对外服端口
        int port = 10100;

        // 逻辑服
        var demoLogicServer = new JsonLogicServer();

        // 启动 对外服、网关服、逻辑服; 并生成游戏业务文档
        SimpleHelper.run(port, List.of(demoLogicServer));
    }
}

监控admin功能

目前有提供现成的,可以监控broker的后台吗?比如能看到有几台broker, 连接了几个逻辑服,每个broker负载如何,还有其他的性能指标等等。

阿里云 SAE 部署 Demo 时,启动后出现 BalancedManager.remove() 方法出现 NullPointerException 异常

see : https://gitee.com/iohao/iogame/issues/I6CS9A

问题场景描述:
阿里云 SAE 部署 Demo 时,启动后出现 BalancedManager.remove() 方法出现 NullPointerException 异常。

image

此时查看/logs/game/game_other.log会发现一直有100开头的ip在断开连接

2023-02-03 15:20:05.811 INFO [Bolt-conn-event-executor-18-thread-751]c.i.game.bolt.broker.server.processor.connection.ConnectionEventBrokerProcessor.onEvent:59 -通知客户端发送模块信息 ConnectionEvent remoteAddress : 100.127.1.194:52642
2023-02-03 15:20:05.812 INFO [Bolt-conn-event-executor-18-thread-751]c.i.g.b.broker.server.processor.connection.CloseConnectionEventBrokerProcessor.onEvent:54 -连接关闭 remoteAddress : 100.127.1.194:52642

问题照成的影响:

  1. 通过在线WebSocket连接测试发现可以正常连接不影响
  2. logs下的日志文件快速增长,长时间会有磁盘空间占满的隐患
  3. 影响正常日志查看
  4. ……

问题的原因与处理:
在 SAE上 做 SLB 的公网端口映射时,应将暴露的 WebSocket 的映射协议设置为 HTTP,旧时代的 SLB 默认似乎不支持 WebSocket

关注 ioGame 的游戏服务器开发者持续增多,2022-09 ~ 至今各月的统计数据

关注 ioGame 的游戏服务器开发者持续增多,2022-09 ~ 至今各月的统计数据;

这里的统计信息是关于开发者关注 ioGame 框架相关的,从统计数据中可以看出,由于 ioGame 上手简单,功能强大等优点,得到了众多开发者的关注。如果你想知道 ioGame 有没有人在使用,可以先到这里看下统计数据、开发者的评价与讨论。

https://www.yuque.com/iohao/game/gpxk93#TwVa8

这里展示了每月的统计数据,统计数据来源于语雀后台,这些数据都是真实的、客观存在的、活的

因为成本的原因,某宝某多还没有出现能提供这种服务的商家,所以这样的统计数据也更具真实性。

通过统计数据,我们可以看到每天会有很多开发者在访问 ioGame 的在线文档,并且这些统计数据不是来源于口嗨的,也不是主观创造的。

所以,还在犹豫要不要使用 ioGame 的开发者们,更应该讨论的是“为什么这些开发者会选择使用 ioGame”,而不是 ioGame 有没有人在使用的问题。

点击我,到语雀后台查看 ioGame 的数据

image

关于高并发

你好!一个非常伟大的开始!
请问客户端高并发访问游戏服务器中的某个方法时,是游戏框架内部有相应的机制去保持原子性还是都需要自己去考虑这些,谢谢!

用户动态绑定逻辑服节点,实现LOL、王者荣耀匹配后动态分配房间节点

支持对外服的玩家绑定指定的游戏逻辑服(可以做到动态分配游戏逻辑服资源)

描述
支持对外服的玩家绑定指定的游戏逻辑服id,如果用户绑定了指定的游戏逻辑服id,之后与该游戏逻辑服的请求都由这个绑定的游戏逻辑服来处理

场景举例

  1. 什么意思呢?这里用匹配与象棋的场景举例。
  2. 假设我们部署了 5 台象棋逻辑服,在玩家开始游戏之前。我们可以在匹配服中进行匹配,当匹配逻辑服把A、B两个玩家匹配到一起了。
  3. 此时我们可以通过 访问【同类型】的多个逻辑服方法,当得到象棋房间数最少的象棋逻辑服后(这里假设是房间数最少的象棋逻辑服是《象棋逻辑服-2》),把《象棋逻辑服-2》的逻辑服id 绑定到 A、B 两个玩家身上。
  4. 之后与象棋相关的操作请求都会由《象棋逻辑服-2》这个游戏逻辑服来处理,比如:开始游戏、下棋、吃棋、和棋等。
  5. 也可以简单点把这理解成,类似 LOL、王者荣耀的匹配机制。在匹配服匹配到玩家后,把匹配结果中的所有玩家分配到一个房间(节点)里面。
  6. 这是一种动态分配资源最少的节点(逻辑服)的用法之一。
  7. 这个版本先做成只能绑定一个逻辑服的,因为暂时没有想到多个的场景。

大概简图如下
输入图片说明

🐳 玩家A和玩家B可以在不同在对外服上,两个玩家发起一个匹配请求。由匹配逻辑服来处理,假设是房间数最少的象棋逻辑服是《象棋逻辑服-2》,那么《象棋逻辑服-2》的逻辑服id 绑定到 A、B 两个玩家身后。之后的处理的简图如下:

输入图片说明

  • 之后与 象棋相关的操作请求 都会由《象棋逻辑服-2》这个游戏逻辑服来处理,比如:开始游戏、下棋、吃棋、和棋等。
  • 也可以简单点把这理解成,类似 LOL、王者荣耀的匹配机制。在匹配服匹配到玩家后,把匹配结果中的所有玩家分配到一个房间(节点)里面。
  • 这是一种动态分配资源最少的节点(逻辑服)的用法之一。

action 业务参数与返回值增加 List 支持

大概使用方式如下

/** 请求 */
@ProtobufClass
@FieldDefaults(level = AccessLevel.PUBLIC)
public class HelloReq {
    String name;
}

@ActionController(6)
public void ListAction {
    @ActionMethod(60)
    public List<HelloReq> list2list(List<HelloReq> helloReqList) {
        
        HelloReq newHelloReq = new HelloReq();
        newHelloReq.name = "hello list, I'm here ";

        List<HelloReq> newList = new ArrayList<>();
        newList.add(newHelloReq);

        return newList;
    }
}

游戏对外服增加路由访问验证、路由权限,将部份逻辑做成单独的类

将 ExternalBizHandler 中路由验证部分独立出来

protected void channelRead0(ChannelHandlerContext ctx, ExternalMessage message) {
        UserSession userSession = UserSessions.me().getUserSession(ctx);

        // 是否可以访问业务方法(action),true 表示可以访问该路由对应的业务方法
        boolean pass = ExternalGlobalConfig.accessAuthenticationHook.pass(userSession, message.getCmdMerge());

        // 当访问验证没通过,通知玩家
        if (!pass) {
            message.setResponseStatus(ActionErrorEnum.verifyIdentity.getCode());
            message.setValidMsg("请先登录,在请求业务方法");
            // 响应结果给用户
            Channel channel = userSession.getChannel();
            channel.writeAndFlush(message);
            return;
        }
... 省略部分代码
}

原生pb与框架互通示例

新增:原生 PB 与 jprotobuf 互转示例,具体查看示例源码的 spring-websocket-native-pb-client 模块
示例文档:https://www.yuque.com/iohao/game/ruaqza

spring-websocket-native-pb-client 模块,单独互转示例代码 NativeProtoTest.java

网络测试步骤:

  1. 启动综合示例 SpringGameOneApplication.java
  2. 启动原生pb 模拟客户端 SpringWebsocketNativeProtoClient.java

SpringWebsocketNativeProtoClient 的请求是使用纯原生的 proto 代码发起请求数据的。

游戏对外服独立 UserSession 管理部份逻辑,做成单独的类

将 ExternalBizHandler 中 UserSession 部份独立成单独的 ChannelInboundHandler

@ChannelHandler.Sharable
public class ExternalBizHandler extends SimpleChannelInboundHandler<ExternalMessage> {
    ... ... 省略部分代码
    @Override
    public void channelInactive(ChannelHandlerContext ctx) {
        // 从 session 管理中移除
        UserSession userSession = UserSessions.me().getUserSession(ctx);
        UserSessions.me().removeUserSession(userSession);
    }

    @Override
    public void channelActive(ChannelHandlerContext ctx) {
        // 加入到 session 管理
        UserSessions.me().add(ctx);
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
        // 从 session 管理中移除
        UserSession userSession = UserSessions.me().getUserSession(ctx);
        UserSessions.me().removeUserSession(userSession);
    }
}

支持 javax.validation 校验

from https://gitee.com/iohao/iogame/issues/I5SLGJ

由于原来老的业务服务都使用的是javax.validation 做数据校验,而不是 jakarta.validation;在整合服务时会出现不兼容的情况,查看源码 ValidatorKit.java 中确实使用的是jakarta.validation, 所以可以考虑优化 ValidatorKit类,我们可以按需选择使用javax.validation 还是jakarta.validation

缩小打 jar 包后的包体,打 jar 包后仅 15MB

使用 ioGame 开发的项目,打 jar 包后的包体大小约 15MB。

移除一些第三方库
为缩小打包,将 hutool 依赖移除、将 fastjson2 依赖配置中的 scope 改为 compile;如有使用到相关的,需要开发者自行引入。

如果你的项目中有使用到 hutool 、fastjson2 的,需要在 pom 中添加

<!-- https://mvnrepository.com/artifact/cn.hutool/hutool-all -->
<dependency>
    <groupId>cn.hutool</groupId>
    <artifactId>hutool-all</artifactId>
    <version>5.8.11</version>
</dependency>

<!-- https://mvnrepository.com/artifact/com.alibaba.fastjson2/fastjson2 -->
<dependency>
    <groupId>com.alibaba.fastjson2</groupId>
    <artifactId>fastjson2</artifactId>
    <version>2.0.23</version>
</dependency>

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.