bigworld源码分析(2)—— loginApp分析
loginApp是整个bigworld进行用户认证的服务,是用户进入游戏的第一步。本篇主要针对loginApp的认证流程,如何和其他服务进行交互,以及loginApp针对多服务负载的不同做法进行分析。
1. loginApp用户认证流程
用户通过loginApp认证,主要是以下几个步骤:
(1) client发送认证请求(前提是他已经知道loginApp的ip和端口)
(2) loginApp在其监听端口上,收到这个请求
(3) 收到请求之后,loginApp将认证信息转发给DBMgr,通过DBMgr来验证登陆认证是否合法
(4) DBMgr通过查询数据库来进行确认
(5) 如果认证是合法的,那么DBMgr通知BaseAppMgr去创建一个新的entity(玩家实例)
(6) BaseAppMgr将创建新entity的需求,转发给当前负载最小的BaseApp
(7) BaseApp创建一个新的Proxy,Proxy就是BaseApp上代表用户的实例
(8) 同时可能需要在CellApp中创建一个玩家entity,这个要根据实际的项目需要了。也有可能直到用户选取了角色之后,才在cellApp中创建entity。
(9) 当上门的事情都做完之后,proxy的UDP端口将结果返回给client(通过BaseAppMgr,DBMgr,最后LoginApp)
上述流程的示意图是这样的:

2. loginApp认证流程在代码中的体现
代码只截取认为最重要的部分和关键语句,大部分代码省略,使用//......来代替
(2)(3)loginApp收到认证请求,并且转发:
// 先针对参数有效性进行检查
// ...... // 判断是不是上一个认证请求还没做完
// First check whether this is a repeat attempt from a recent pending
// login before attempting to decrypt and log in.
if (this->handleResentPendingAttempt( source, header.replyID ))
{
// ignore this one, it's in progress
loginStats_.incPending();
return;
} // 判断是否是否了加密
#ifdef USE_OPENSSL
Mercury::PublicKeyCipher * pPrivateKey = &privateKey_;
#else
Mercury::PublicKeyCipher * pPrivateKey = NULL;
#endif //.......
// 判断是否之前有认证cache
// First check whether this is a repeat attempt from a recent
// resolved login before attempting to log in.
if (this->handleResentCachedAttempt( source, pParams, header.replyID ))
{
// ignore this one, we've seen it recently
return;
} // ...... INFO_MSG( "Logging in %s{%s} (%s)\n",
pParams->username().c_str(),
pParams->password().c_str(),
source.c_str() ); // Remember that this attempt is now in progress and discard further
// attempts from that address for some time after it completes.
cachedLoginMap_[ source ].reset();
cachedLoginMap_[ source ].pParams( pParams ); DatabaseReplyHandler * pDBHandler =
new DatabaseReplyHandler( source, header.replyID, pParams );
// 发送给DBMgr
Mercury::Bundle & dbBundle = this->dbMgr().bundle();
dbBundle.startRequest( DBInterface::logOn, pDBHandler ); dbBundle << source << false /*off channel*/ << *pParams; this->dbMgr().send();
(9) 认证消息返回给client代码
void DatabaseReplyHandler::handleMessage( const Mercury::Address & /*source*/, Mercury::UnpackedMessageHeader & header, BinaryIStream & data,
void * /*arg*/ )
{
uint8 status;
data >> status;
// 检查认证返回码
if (status != LogOnStatus::LOGGED_ON)
{
// 如果认证失败
// ......
} LoginReplyRecord lrr;
data >> lrr; // If the client has an external address, send them the firewall
// address instead! if (!LoginApp::instance().netMask().containsAddress( clientAddr_.ip ))
{
INFO_MSG( "DatabaseReplyHandler::handleMessage: "
"Redirecting external client %s to firewall.\n",
clientAddr_.c_str() );
lrr.serverAddr.ip = LoginApp::instance().externalIP();
}
// 返回client认证结果,并且缓存认证结果
LoginApp::instance().sendAndCacheSuccess( clientAddr_, replyID_, lrr, pParams_ ); gNumLogins++; delete this;
}
3. loginApp是如何做负载的
loginApp是bigworld中用来处理用户登录认证的,为了防止单一loginApp存在性能瓶颈,同时防止单点故障,bigworld也是可以运行同时运行多个loginApp。不过我们知道多个baseApp和cellApp是分别有,baseAppMgr和cellAppMgr来做管理和负载的,loginApp我们却没有找到类似的loginAppMgr。
在bigworld的文档中,是这样写的:
In terms of load, a single LoginApp should be sufficient for most purposes. However, as this is still a potential bottleneck, and single point of failure, BigWorld supports running multiple LoginApps.
The remaining issue then is how to distribute the client login requests across multiple login servers. The standard way to do this is use a DNS solution similar to how popular web sites balance traffic load across multiple machines.
从上面的介绍来看,bigworld本身确实是没有提供类似的loginAppMgr,而是需要我们使用DNS来做负载,DNS返回不同的loginApp ip,来分担压力。
bigworld源码分析(2)—— loginApp分析的更多相关文章
- bigworld源码分析(3)——dbMgr分析
dbMgr主要是玩家数据的读取和保存的,例如在bigworld源码分析(3)中,玩家在认证的时候,loginApp需要通过dbMgr来验证玩家数据是否合法,这就是针对玩家的账号数据进行查询.本篇中,我 ...
- bigworld源码分析(1)—— 研究bigworld的意义和目标
对于网络游戏服务器开发熟悉的人,基本都知道bigworld引擎,此引擎包括客户端部分和服务器部分,已经有很多知名的网络游戏通过bigworld来构建游戏.我主要关注bigworld的服务器部分,它是一 ...
- ArrayList源码和多线程安全问题分析
1.ArrayList源码和多线程安全问题分析 在分析ArrayList线程安全问题之前,我们线对此类的源码进行分析,找出可能出现线程安全问题的地方,然后代码进行验证和分析. 1.1 数据结构 Arr ...
- Okhttp3源码解析(3)-Call分析(整体流程)
### 前言 前面我们讲了 [Okhttp的基本用法](https://www.jianshu.com/p/8e404d9c160f) [Okhttp3源码解析(1)-OkHttpClient分析]( ...
- Okhttp3源码解析(2)-Request分析
### 前言 前面我们讲了 [Okhttp的基本用法](https://www.jianshu.com/p/8e404d9c160f) [Okhttp3源码解析(1)-OkHttpClient分析]( ...
- Spring mvc之源码 handlerMapping和handlerAdapter分析
Spring mvc之源码 handlerMapping和handlerAdapter分析 本篇并不是具体分析Spring mvc,所以好多细节都是一笔带过,主要是带大家梳理一下整个Spring mv ...
- HashMap的源码学习以及性能分析
HashMap的源码学习以及性能分析 一).Map接口的实现类 HashTable.HashMap.LinkedHashMap.TreeMap 二).HashMap和HashTable的区别 1).H ...
- ThreadLocal源码及相关问题分析
前言 在高并发的环境下,当我们使用一个公共的变量时如果不加锁会出现并发问题,例如SimpleDateFormat,但是加锁的话会影响性能,对于这种情况我们可以使用ThreadLocal.ThreadL ...
- 物联网防火墙himqtt源码之MQTT协议分析
物联网防火墙himqtt源码之MQTT协议分析 himqtt是首款完整源码的高性能MQTT物联网防火墙 - MQTT Application FireWall,C语言编写,采用epoll模式支持数十万 ...
- Netty 源码学习——客户端流程分析
Netty 源码学习--客户端流程分析 友情提醒: 需要观看者具备一些 NIO 的知识,否则看起来有的地方可能会不明白. 使用版本依赖 <dependency> <groupId&g ...
随机推荐
- SQL触发器实例
SQL触发器实例讲解(本文是来自百度文库) 备注:本人建了一个站特价汇,我想记录每个商品的点击量,然后按照点击量来牌名商品,想要提高效率,所以必须得用触发器,下面是本人在百度文库中的找到的学习资料,分 ...
- opengl视图变换 投影变换推导
视图变换在opengl中,视图变换的输入是:(1)眼睛位置(或者说相机位置)eys:(2)眼睛朝向的中心center,(就是眼睛朝哪里看);(3)头的方向up.任何一点经过视图变换后都会转化到眼睛坐标 ...
- Less入门到上手——前端开发利器<二>深入了解
接着昨天的继续... ... 4.嵌套: HTML部分 <table> <tr> <th colspan="3">测试列表标题</th&g ...
- 【ros】.bag文件
Bags are typically created by a tool like rosbag They store the serialized message data in a file as ...
- mysqlbinlog flashback 5.6完全使用手册与原理
版本更新 2016/3/7 首次发布 2016/3/9 修复update无效问题 2016/4/8 修复带有秒级别以下的日期数据bug 下载地址 http://pan.baid ...
- hello word
开通微博,用于记录在工作中遇到的点滴问题. 2015/08/31
- Retrofit 2.0使用(2)如何使用@Body的形式发送Post
在使用Retrofit的时候如果只是有几个参数我们可以用@Querry的形式,然后需要使用','隔开 但是在需要@Querry的参数多了之后,如果再用上面的方式就会造成参数写了一大堆的麻烦事 所以Re ...
- KindEditor4.1.10,支持粘贴图片(转载!)
本人扩展了KindEditor4.1.10,使得他能够在Chrome和IE11中直接粘贴复制的图片(比如通过截图工具把图片直接保存在剪切板中),然后调用上传URL上传图片 方法,修改kindedito ...
- WEB项目 后台接收前端数组
//保存区域选择的设备 $scope.saveDevice = function(){ var device = []; $("input[type='checkbox']:checked& ...
- sentinel搭建redis集群经验总结
一.protected-mode默认情况下,redis node和sentinel的protected-mode都是yes,在搭建集群时,若想从远程连接redis集群,需要将redis node和se ...