ejabberd源码流程梳理
ejabberd的工程主要通过ejabberd.app 组织起来
ejabberd.erl :
application:start(ejabberd).
ejabberd_app.erl:
Mod:start(normal, _Args); 该函数中启动ejabberd_listener:start_listeners()
ejabberd_listener.erl
通过 conf 得到<!---->ejabberd_c2s,
开始listen
如果有连接,accpet之后
<!---->ejabberd_socket:start(Module, gen_tcp, Socket, Opts)
Receiver=ejabberd_receiver:start(
Socket,SockMod,none,MaxStanzaSize),
ejabberd_receiver.erl
ejabberd_receiver是一个gen_server进程(behaviour),
receiver 收到数据(handle_info)发送到xml_stream.
xml_stream解析数据后发送到c2s状态机.
ejabber_c2s.erl
是一个gen_fsm(有限状态机),维护一系列client和server之间的状态。gen_fsm的状态改变是由gen_fsm:send_event(FsmRef,Event)触发的。在ejabberd里,ejabberd_receiver通过xml_stream把事件和xml发送至ejabber_c2s改变其状态。
ejabber_c2s有限状态机的初始状态是wait_for_stream,其余的状态有:
wait_for_auth,wait_for_feature_request,wait_for_sasl_response,wait_for_bind,wait_for_session,session_established
ejabberd官方文档中提到的那些关于xmpp收到的事件的hook,就是在c2s中实现的。
c2s中调用ejabberd_hooks:run_fold()来运行相应的hook。
1.Presence消息流程
用户登陆过程中:
在c2s的waitforsession中调用
{Fs, Ts} = ejabberd_hooks:run_fold(
roster_get_subscription_lists,
StateData#state.server,
{[], []},
[U, StateData#state.server]),
获取from、to好友列表,分别置入state. pres_f和state. pres_t中。
客户端登陆成功后,给服务器发送
<presence id="B0DjG-12">
<status>在线</status>
<priority>1</priority>
</presence>
该presence没有目的地址(to)。
服务端收到该消息在c2s的session_established2中处理,
调用presence_update(FromJID, PresenceEl,StateData)
èpresence_broadcast_first(From, NewStateData, Packet);
(1) 给state.pres_t中每个成员发送probe;
(2)给state.pres_f中每个成员发送presence
2.添加好友流程
以Lihengz 添加 admin 好友为例
下列状态值中第一项为“订阅(subscription)”,第二项为“挂起(ask)”
Lihengz的状态变化为 :out_state_change(none,none, subscribe) -> {none, out};
Admin的状态变化为:in_state_change(none,none, subscribe) -> {none, in};
Admin接受请求后,状态变化:out_state_change(none,in, subscribed) -> {from, none};
Lihengz状态变化:in_state_change(none,out, subscribed) -> {to, none};
如果admin也加lihengz为好友:
admin状态变化:out_state_change(from,none, subscribe) -> {from, out};
lihengz状态变化: in_state_change(to, none, subscribe) -> {to, in};
lihengz接受请求后,状态变化:out_state_change(to, in, subscribed) -> {both, none};
admin状态变化:in_state_change(from,out, subscribed) -> {both, none};
3,在服务器上添加通讯录
在服务器上为mine添加mine1为好友(mine1未设置自动添加好友)
mine客户端会收到如下消息:
<iq id="push3140513066"to="mine@xmppserver/Spark 2.6.3" from="mine@xmppserver"type="set">
<query xmlns="jabber:iq:roster">
<item jid="mine1@xmppserver" subscription="none"ask="subscribe"/>
</query>
</iq>
如果mine1拒绝请求,mine客户端就收不到后续消息了。Roster表中一直维持挂起状态。
如果mine1设置为自动添加好友,mine客户端会收到:
<iq id="push3548743800"to="mine@xmppserver/Spark 2.6.3" from="mine@xmppserver"type="set">
<query xmlns="jabber:iq:roster">
<item jid="mine1@xmppserver" subscription="both"/>
</query>
</iq>
from: http://www.cnblogs.com/unqiang/p/4402418.html
ejabberd源码流程梳理的更多相关文章
- Eureka服务端源码流程梳理
一.简述 spring cloud三步走,一导包,二依赖,三配置为我们简化了太多东西,以至于很多东西知其然不知其所以然,了解底层实现之后对于一些问题我们也可以快速的定位问题所在. spring clo ...
- Eureka客户端源码流程梳理
前面梳理了Eureka服务端的流程,现在整理下客户端的流程. 1.在这个包(spring-cloud-netflix-eureka-client)里面寻找客户端启动入口相关配置,关键配置文件sprin ...
- ClickHouse源码笔记5:聚合函数的源码再梳理
笔者在源码笔记1之中分析过ClickHouse的聚合函数的实现,但是对于各个接口函数的实际如何共同工作的源码,回头看并没有那么明晰,主要原因是没有结合Aggregator的类来一起分析聚合函数的是如果 ...
- Flask 源码流程,上下文管理
源码流程 创建对象 from flask import Flask """ 1 实例化对象 app """ app = Flask(__na ...
- ES6.3.2 index操作源码流程
ES 6.3.2 index 操作源码流程 client 发送请求 TransportBulkAction#doExecute(Task,BulkRequest,listener) 解析请求,是否要自 ...
- django-admin的源码流程
一.admin的源码流程 首先可以确定的是:路由关系一定对应一个视图函数 a.当点击运行的时候,会先找到每一个app中的admin.py文件,并执行 b.执行urls.py admin.site是什么 ...
- Django session 源码流程
流程 Django session源码流程 首先执行的是SessionMiddleware的init方法 import_module(settings.SESSION_ENGINE) 导入了一个 dj ...
- rest_framework解析器组件源码流程
rest_framework解析器组件源码流程 解析器顾名思义就是对请求体进行解析.为什么要有解析器?原因很简单,当后台和前端进行交互的时候数据类型不一定都是表单数据或者json,当然也有其他类型的数 ...
- 【转】ANDROID自定义视图——onMeasure,MeasureSpec源码 流程 思路详解
原文地址:http://blog.csdn.net/a396901990/article/details/36475213 简介: 在自定义view的时候,其实很简单,只需要知道3步骤: 1.测量—— ...
随机推荐
- Keil V4.72升级到V5.1X之后
问题描述 Keil V4.72升级到V5.1x之后,原来编译通过的工程,出现了如下错误: .\Libraries\CMSIS\CM3\DeviceSupport\ST\STM32F10x\STM32f ...
- EasyUI + EF + MVC4 后台截图
到目前完成的页面截图,完成了增删改查几项功能的技术测试,在解决几个小问题,就重新设计结构开始一个完整的后台开发,坚持用博客和云笔记记录开发过程.
- 基于SuperSocket实现的WebSocket(后端)
关于WebSocket其实很早就想发了,奈何之前项目中的WebSocket的后端不是我做的,而我又想前后端都发出来和大家讨论讨论~于是挤出点时间研究了一下WebSocket的后端实现(所以才有了这篇文 ...
- js获取当前浏览器页面高度及宽度信息的方法
var scrollLeft = Math.max(document.documentElement.scrollLeft, document.body.scrollLeft); var scroll ...
- Jquery实现自动提示下拉框
1.引入脚本库: <script type="text/javascript" src="/Jscripts/jquery-1.3.2.js"&g ...
- linux驱动系列之makefile
在linux环境下做嵌入式无论是编写应用程序还是驱动程序等等,都需要用make来进行程序的编译,就需要学会自己编写Makefile.Makefile主要的作用有3点:1.决定编译哪些文件 2.怎样编译 ...
- 360随身wifi怎样购买?360随身wifi怎样预约?
---恢复内容开始--- 360随身wifi怎样购买 想要购买360随身Wifi,可以登录360随身Wifi的官网wifi.360.cn,或者直接登陆京东商城进行购买,售价为19.9元,分黑.白两色. ...
- PAT-乙级-1010. 一元多项式求导 (25)
1010. 一元多项式求导 (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 设计函数求一元多项式的导数.(注:xn(n为整数)的一 ...
- Firefly安装说明 与 常见问题
原地址:http://bbs.gameres.com/thread_223688.html 第三方库依赖: twisted, python-memcached ftp://ftp.tummy.c ...
- [Unity+Android]横版扫描二维码
原地址:http://blog.csdn.net/dingxiaowei2013/article/details/25086835 终于解决了一个忧伤好久的问题,严重拖了项目进度,深感惭愧!一直被一系 ...