python3.7游戏_obEspoir: obespoir 是一款个人独立开发的彻底分布式的开源游戏框架,主要基于python3.7 进行开发,采取了 asyncio,uvloop 等异步编程技术,...

一、项目简介

​obespoir 是一款个人独立开发的彻底分布式的开源游戏框架,主要基于python3.7 进行开发,采取了 asyncio,uvloop 等异步编程技术,保障高性能的同时,从框架层面彻底解决生产环境中单节点或服务的高可用需求问题。

该框架的主要特点如下:

支持动态热更新:通过嵌入 http 接口实现游戏内各类型节点的实时更新,支持配置文件和数据库等多种方式;

自适应高可用:框架中的每一个节点可根据需求动态添加或删除,任何节点发生故障无需停服,框架会自动发现合适的节点维持业务运行;

性能卓越:基于异步 asyncio, uvloop 等优秀三方库开发,并发性能接近于 golang, c等编译型语言, 普通单机并发可达到万数量级;

安全可靠:节点间根据配置进行分层权限管理,有效应对 DDOS 瘫痪全局节点;高效 rpc 通信加密机制,防止数据伪造及篡改;

跨平台:提供 websocket 连接服务,支持客户端 h5 游戏引擎开发,一份代码适用移动端,PC 端,浏览器等各终端。

二、安装部署

1. 使用环境

1. 代码基于python3.7进行开发,请预先装好相应环境;

2. 有使用到三方库ujson, pymongo(如果配置不用mongo可不装)

2. 使用说明

安装步骤:

pip3 install -i https://pypi.python.org/simple obEspoir

引入该框架需要对接一下几处地方:

2.1 配置文件

​配置文件采用json格式,每一个节点(进程)有一个自己的配置文件,配置文件里包括该节点监听的各类端口,日志的目录,连接的数据库地址,连接加密密码、路由信息等等。

​其中rpc接口部分需要配置哪些消息需要转发,又需要转发往哪些节点,支持两种方式::

"route":{//配置消息转发的路由

"range":{

"route":[[0,5000]],"service":[[5001,1000000]]

},

"special":{

"route":[1000]//like"proxy":[110119]

}

}

1.消息ID范围:range中说明哪些范围段内的消息往哪类型节点发,比如上述[0,5000]发往route类型节点(如果本节点就是route类型,则本地处理),[5001,10000]内的消息发往service类型节点;

2.具体的消息ID:special中可以具体写明哪些指定的消息发往哪个节点,如1000消息发往route处理,如果该部分和range冲突,以special配置优先;

总体配置信息参考如下:

proxy类型节点:

{

"log_level":"debug",//日志级别

"name":"proxy_1",//节点名称

"log_dir":"logs/obEspoir/",//项目运行日志存放目录(不同测试环境目录不同)

"host":"127.0.0.1",

"type":"proxy",

"api_path":"proxy.include_libs",//api路径,项目启动时导入自定义的各类api模块的地方

"available_way":"local",//高可用性配置使用哪种方式存储:local(本地文件存储,默认方式),mongo(mongodb存储)

"mongo_uri":"mongodb://test:xxxx@xxxx.mongodb.rds.aliyuncs.com:3717,xxxx.mongodb.rds.aliyuncs.com:3717/admin",//如果使用了MongoDB作为存储,则该项需要配置

"remote_ports":[//需要连接的远程rpc端口信息

{

"host":"127.0.0.1",

"port":21002,

"token":"helloworldiloveyou~1234567890123",

"encode":0,

"type":"route"//节点类型(字符串标识)

}

],

"websocket":{//本地监听的websocket端口配置

"port":20000,

"token":"helloworldiloveyou~1234567890123",//32位秘钥websocket通信

"encode":0,//是否启用加密

"timeout":300,//多久连接超时(单位为秒)

"no_state":{//无状态的消息ID

"range":[[0,100000]],

"special":[]//like"game":[110119]

}

},

"http":{//本地监听的http端口信息

"port":20001,

"token":"helloworldiloveyou~1234567890123",//32位秘钥httpserverweb通信

"encode":0//是否启用加密

},

"rpc":{//本地监听的rpc端口信息

"port":20002,

"token":"helloworldiloveyou~1234567890123",//32位秘钥rpc通信

"encode":0,//是否启用加密

"type":"proxy",//节点类型

"reconnect_time":5//连接断开时重连时间间隔,默认最多重试3次

}

}

route路由类型节点参考配置(大多和proxy相同):

{

"log_level":"debug",//日志级别

"name":"route_1",//项目名称

"log_dir":"logs/obEspoir/",//项目运行日志存放目录(不同测试环境目录不同)

"host":"127.0.0.1",

"type":"route",

"api_path":"route.include_libs",//api路径

"http":{

"port":21001,

"token":"helloworldiloveyou~1234567890123",//32位秘钥httpserverweb通信

"encode":0//是否启用加密

},

"rpc":{

"port":21002,

"token":"helloworldiloveyou~1234567890123",//32位秘钥rpc通信

"encode":0,//是否启用加密

"type":"game",//节点类型

//rpc会收到三类消息:待本地处理消息、待转发的消息、待直接推送到websocketclient的消息

"route":{//配置消息转发的路由

"range":{

"route":[[0,5000]],"service":[[5001,1000000]]

},

"special":{

"route":[1000]//like"proxy":[110119]

}

}

},

"remote_ports":[

{

"host":"127.0.0.1",

"port":20002,

"token":"helloworldiloveyou~1234567890123",

"encode":0,

"type":"proxy"//节点类型(字符串标识)

},

{

"host":"127.0.0.1",

"port":22002,

"token":"helloworldiloveyou~1234567890123",

"encode":0,

"type":"service"//节点类型(字符串标识)

}

]

}

service业务节点参考配置(大多和proxy相同):

{

"log_level":"debug",//日志级别

"name":"service_1",//项目名称

"log_dir":"logs/obEspoir/",//项目运行日志存放目录(不同测试环境目录不同)

"host":"127.0.0.1",

"type":"service",

"api_path":"service.include_libs",//api路径

"http":{

"port":22001,

"token":"helloworldiloveyou~1234567890123",//32位秘钥httpserverweb通信

"encode":0//是否启用加密

},

"rpc":{

"port":22002,

"token":"helloworldiloveyou~1234567890123",//32位秘钥rpc通信

"encode":0,//是否启用加密

"type":"service",//节点类型

//rpc会收到三类消息:待本地处理消息、待转发的消息、待直接推送到websocketclient的消息

"route":{//配置消息转发的路径

"range":{

"service":[[5001,1000000]]

},

"special":{

//like"proxy":[110119]

}

}

},

"remote_ports":[

{

"host":"127.0.0.1",

"port":21002,

"token":"helloworldiloveyou~1234567890123",

"encode":0,

"type":"route"//节点类型(字符串标识)

}

]

}

2.2 自定义接口配置

​实际的应用中我们可以根据需要定制不同节点的接口服务,包括rpc接口消息处理,http接口消息处理,websocket接口消息处理。最后将自定义的三类接口文件在配置文件中的“api_path”对应的文件中导入即可,如:

%E8%87%AA%E5%AE%9A%E4%B9%89%E6%8E%A5%E5%8F%A3%E5%AF%BC%E5%85%A5.jpg

2.2.1 http接口

​每个节点都可以对外提供http服务,收到不同的http服务请求,然后调用不同的处理方法,比如可以做服务端热更新,或者用来不同模块间实时更新用户个人信息。

​参考自定义接口文件http_handler.py写法:

from aiohttp import web

from obespoir.httpserver.route import HttpHandler

from obespoir.server.server import Server

@HttpHandler("/")

async def index(request):

return web.Response(body="hello", content_type="text/html")

@HttpHandler("/update_remote_rpc_config")

async def update_remote_rpc_config(request):

await Server().update_remote_rpc_config()

return web.Response(body="ok~", content_type="text/html")

2.2.2 rpc接口

​每个节点同时也可以对外支持rpc服务,收到不同的rpc调用请求,调用不同的处理方法,实际处理的自定义接口,参考写法如下(rpc_handler.py,文件名和include_libs.py中import路径保持一致即可):

from obespoir.base.common_define import NodeType

from obespoir.base.ob_handler import BaseHandler, RegisterEvent

from obespoir.rpcserver.push_lib import push_message

from obespoir.share.ob_log import logger

@RegisterEvent(100002)

class LoginHandler(BaseHandler):

async def execute(self, *args, **kwargs):

logger.info("login_handler:{} {}".format(args, kwargs))

user_id = self.params.get("user_id", -1)

passwd = self.params.get("passwd", "")

if -1 == user_id or not passwd:

return {}

# ...

pass

return {"code": 200, "data": {}}

@RegisterEvent(100130, need_return=False)

class OfflineHandler(BaseHandler):

async def execute(self, *args, **kwargs):

logger.info("offline: {}, {}".format(args, kwargs))

pass

return {"code": 200}

@RegisterEvent(10000, need_return=True)

class HeartBeatHandler(BaseHandler):

async def execute(self, *args, **kwargs):

logger.info("heartbeat: {}, {}".format(args, kwargs))

pass

return {"code": 200}

2.2.3 websocket接口

​理论上每个节点都可以对外提供websocket服务(实际中一般只有proxy节点才会配置),收到websocket消息请求后,具体的接口处理方法和逻辑由用户自己决定,接口参考写法(ws_handler.py)如下:

import asyncio

import ujson

from websockets.exceptions import ConnectionClosed

from obespoir.share.singleton import Singleton

from obespoir.share.ob_log import logger

from obespoir.base.common_define import NodeType

from obespoir.base.ob_protocol import DataException

from obespoir.base.global_object import GlobalObject

from obespoir.websocketserver.route import webSocketRouteHandle

from obespoir.rpcserver.push_lib import push_message

@webSocketRouteHandle

async def forward_0(command_id, data, session_id):

"""

消息转发

:param command_id: int

:param data: json

:param session_id: string

:return: None

"""

print("forward_0", command_id, data, type(data), data, session_id)

if not isinstance(data, dict):

try:

data = ujson.loads(data)

print("data type :", type(data))

except Exception:

logger.warn("param data parse error {}".format(data))

return {}

data.update({"message_id": command_id})

await push_message(NodeType.ROUTE, command_id, data, session_id=session_id, to=None)

2.3 示例demo

在项目中有一个test文件目录,其中为简单的测试demo,启动时执行以下文件即可

服务端启动:

python3 start_route.py # 启动一个route类型节点

python3 start_proxy.py # 启动一个proxy类型节点

python3 start_service.py # 启动一个service类型节点

测试客户端:

python3 test_frame.py # 客户端测试文件

三、功能介绍

设计原则: 拟计划设计出一款彻底分布式的,去中心化的分布式app后端引擎架构。

框架功能:封装基础通信父类,用户只需要修改配置文件,如果需要自定义,继承相应父类即可;

1. 接口设计

1. webport: http接口

2. websocket port: 长连接接口: 供客户端使用 --- ws port

3. rpc port: rpc连接端口: 供其他进程rpc调用

4. remote port: 启动时希望去连接的其他节点地址和端口

obEspoir框架后的总体分布式架构图如下

kuangjia.jpg

​client: 客户端,即玩家用户,游戏中客户端和服务端之间的连接是长连接,客户端和服务端的proxy节点进行连接;

​proxy:服务端的代理节点,其主要任务是负责消息打包和解包,加解密,然后将合法的消息转发向后端节点。proxy节点地址对公网开发(客户端通过域名和端口连接);

​route: 服务端的消息分发节点(路由节点)。该节点根据消息请求id将不同的消息分发到不同的子节点中进行处理,如长沙麻将分发到长沙麻将的游戏节点处理,广东麻将分发到广东麻将节点处理,该类型节点一般不对公网开放;

​service: 游戏节点,各类型的游戏,如csmj(长沙麻将),gdmj(广东麻将), xzmj(血战)等。此类节点为游戏主逻辑节点。

2. 自动高可用性

​该框架中节点一般分为三类型(可根据用户需求自行扩展):proxy,route, service. 节点之间互不依赖, 支持分布式架构,任何一个节点故障不会影响到整体业务运行。

2.1 节点类型

框架本身里面所有的节点本质都是一样的,只是赋予了不同节点不同类型的职能。初步职能如下:

proxy代理节点: (对公网开放,不承担消息转发功能)

客户端通过websocket端口连接某一个代理节点

route路由节点: (只对内网开放)

负责节点间消息转发,当其他节点不知道消息如何走向为最短路径时,消息发往路由节点进行中转;

管理记录各节点的存活状态,响应节点相关请求

service业务节点: (只对内网开放)

负责处理各类业务逻辑

2.2 节点标识

节点id(节点id=md5(host+rpc_port))

节点名称(同一台机器上节点名称不能一样)

节点类型:目前分为三种,可自定义

节点4个端口: http_port, websocket_port, rpc_port, remote_ports

节点所在主机标识(主机标识+节点名 唯一定位一个具体的节点)

路由节点地址

2.3 节点热更新策略

​新节点加入时:读取配置,尝试连接配置中其需要rpc连接的所有节点, 失败则重试若干次;同时着重通知当前所有的route节点,有新的存活的节点加入;

​路由节点加入:路由节点需要和所有其他普通节点间保持双向rpc连接,收到其他节点建立连接请求时:鉴权正确,则建立连接;

​未知原因和其他节点主动进行的rpc连接断开时:隔指定周期询问route节点断开连接的另一端节点是否存活,若存活则重新尝试连接;

3. 消息通信

3.1 消息流程

a. 客户端主动请求示例消息:

client发送消息--> proxy节点websocket port收到消息--> rpc调用后续service节点rpc端口 --> service消息处理完后发送消息给proxy

b. service发送消息给proxy时,如果没有直接连接,则发往路由节点route.

c. 服务端推送消息时:

service发送推送消息时,如果没有proxy节点的直接连接,则同样发往路由节点route.

d. route路由节点:

rpc端口收到消息时,判断消息的目的节点:

目的节点为自己: 执行自我业务逻辑处理

目的节点为其他: 向目的节点的rpc端口发送消息

e. proxy 节点:

1. rpc端口收到消息时:

直接调用websocket连接向客户端发送消息

2. 向远程的route或其他节点rpc发送消息时:

如果没有明确发送目标,则发往路由节点

有明确的目标类型,则随机发往一个目标类型的实例化节点

f. service 节点:

判断消息的目的节点:

目的节点为自己: 执行自我业务逻辑处理

目的节点为其他: 向目的节点的rpc端口发送消息

3.2 route路由节点

​存放所有节点的路由信息,当系统中更新或删除节点时,更新路由节点,然后路由节点通知所有的其他节点

3.3 proxy代理节点

1. 连接序号生成:

每次新增连接时产生一个新的连接序号,每次连接和一个连接序号相对应,连接序号和连接会缓存起来

2. websocket收到消息后,将合法消息发往route节点

消息内容有:

session_id: proxy节点id+连接序号

to: 优先使用上一次去往的节点(如果上一次的节点不可用,则等待一段时间再试,三次后放弃请求,连接断开,连接断开时,清空缓存),为空值时,代表未知消息在哪里处理,此时由转发到的下一个节点决定,职位节点id

3. proxy节点需要缓存上一次消息通信的路径(即是否之后的消息都沿用之前消息处理的路径)

消息分为两种类型: 有状态(下一个请求和上一个请求必须同一个service节点处理,默认有状态)和无状态(无需同一个service节点处理不同请求)

3.4 service业务节点

​该节点用来处理具体的业务逻辑,业务处理完后,调用相应的推送接口,向客户端返回消息,其中session_id中包括了该消息是由哪一个proxy发送过来,proxy中的哪一个websocket连接。

4. rpc消息

4.1 rpc消息结构

消息目的地: 客户端过来的消息,即proxy的消息第一次时不知道去往哪里,此时proxy会将消息目的地置为route节点

消息来源:该消息的原始节点名称

消息内容:

route根据消息ID, 决定接下来是否需要rpc请求不同的service节点

rpc连接建立时,验证token信息,通过后后续的请求才会被处理,验证失败则连接断开;

rpc通信时无需加密解密操作,但需要有打包解包(防止粘包现象)

4.2 rpc消息处理流程

​某个节点发送rpc消息时:

根据两个参数判断: next_node_type(发送往下一个节点的类型), to(最终的目的节点id)

a. next_node_type类型和自己相同,返回错误

b. 存在to对应的rpc连接(活跃状态),直接通过该连接发送消息

c. next_node_type存在,选择一个活着的next_node_type类型的实例,发送消息

若没有,发往一个route节点;

若route节点无活跃连接,该次暂停消息发送

​接收rpc消息时:

某个节点收到rpc消息时,会收到两个参数 session_id(消息最初来源于谁, proxy节点id+连接序号), to(消息最终发往谁)

a. 如果to有具体目标,同时该目标不是自己,并且有和该目标的直接rpc连接,则转发请求到目标节点

b. 如果to有具体目标,同时该目标是自己, 则进入本节点消息处理逻辑;

c. 如果to无具体目标:

如果节点为proxy类型节点(终端节点),则调用websocket向客户端发送消息

结合配置根据消息ID判断下一步流程:

有路由走路由,调用发送rpc消息接口

没有路由,本地消息处理:进入本节点消息处理

5. 客户端断线流程

服务端的检测到用户断线时, 会触发以下流程:

清理proxy节点会话缓存

向route发送断线消息

route清理自己的缓存

向后端节点发送断线消息

后端特定service节点收到断线请求后,触发断线逻辑处理

...

6. 配置更新

节点配置信息支持两种方式:

a. 本地文件存储

b. mongodb数据库存储

配置更新:节点对外开放一个web http服务接口,接口收到消息后,更新配置信息

7. 代码结构

base/: 框架中抽象出来的基类库

common_define.py: 常量定义,包括节点类型,连接状态等

data_pack.py: 通信时数据的打包和解包类封装

global_object.py:全局对象,封装进程中多个地方需要用到的变量信息

ob_handler.py: 异步消息实际处理的自定义基类

ob_protocol.py: 异步消息处理的协议

ob_route.py: 消息分发的基类

db/: 数据存储相关,目前暂时只扩展了MongoDB

mongo.py: 使用MongoDB存储配置信息,封装相应方法

httpserver/: http请求相关

handler.py: 不同http请求实际消息处理接口,用户需求不同,该文件实现即不同,此处只为参考

route.py: http请求路由封装类

rpcserver/: rpc请求相关

connection_manager.py: rpc连接管理类

handler.py:不同rpc消息实际处理接口,用户需求不同,该文件实现即不同,此处只为参考

protocol.py: 接收rpc请求的消息处理协议

push_lib.py: 向其他节点推送rpc消息封装类

push_protocol.py: 推送rpc秦秋的消息处理协议

route.py:rpc消息路由封装类

session_cache.py: 长连接会话缓存处理类

server/: 服务主入口

server.py: 框架启动的主入口类

share/: 项目公用工具库

encodeutil.py: 加密解密封装类,消息通信时会调用

ob_log.py: 自定义日志类

parse_json.py: 带注释的json文件解析封装类

singleton.py: 线程安全的单例基类封装

websocketserver:

handler.py: websocket请求的实际消息处理接口,用户需求不同,该文件实现可不同,此处只为参考

manager.py: websocket连接管理类

protocol.py: websocket通信协议封装类

route.py: websocket消息路由封装类

四. 后续

​限于时间原因,obEspoir尚在陆续完善过程中,接下来的主要优化方向:

1. 进一步提高安全性,添加防恶意攻击篡改和重放攻击等;

2. 用户体验,简化配置模块,使用户更易于操作;

3. 修复可能存在的bug;

4. 未完待续...

​目前已有基于该obEspoir框架开发的多人自定义麻将代码,地址在:

​联系方式:

​微信: jamonhe

weixin.jpeg

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值