副标题#e#
参考 http://www.cnblogs.com/yjl49/archive/2011/09/02/2371964.html
1.客户端连接服务器后,发送消息给服务器的xml数据处理在ejabberd_c2s中。
?? ejabberd_c2s 是一个gen_fsm 状态机,在socket连接建立之后创建。
?? 初始状态为 wait_for_stream,接受形为 {xmlstreamstart,_Name,Attrs} 的消息,其他任何消息都会导致stop。
?? 接受到连接请求,发送feature 请求后状态改为wait_for_feature_request 。
?? 接受到feature 再发送challenge 后,状态改为 wait_for_sasl_response。
?? 经过sasl鉴权后,状态改为 wait_for_stream, 此时 StateData#state.authenticated 已经不为false。
?? 客户端重新发起<stream> 连接请求,服务器发送bind 消息,状态改为 wait_for_bind 。
?? 客户端发送bind 与资源绑定,服务器按照策略验证是否允许相同的用户用不同的资源连接,通过后状态改为 wait_for_session。
?? 客户端发送<iq> 消息创建session,服务器调用 ejabberd_sm:open_session 后将状态改为 session_established。
?? 客户端发送普通的通信消息<iq>,<presence>,<message> 等,都通过session_established2/2 函数处理。
- %%?Process?packets?sent?by?user?(coming?from?user?on?c2s?XMPP??
- %%?connection)??
- session_established2(El,?StateData)?->??
- ??
- ????%%从变量El中取出Name,Attrs?两个参数,El必须是一个以xmlelement?为第一个元子的元组??
- ????{xmlelement,?Name,?Attrs,?_Els}?=?El,??
- ????%%从StateDate中取出user,server,jid???
- ????User?=?StateData#state.user,??
- ????Server?=?StateData#state.server,??
- ????FromJID?=?StateData#state.jid,??
- ??
- ????%%从Attrs属性变量中获取to(也就是发送给谁)??
- ????To?=?xml:get_attr_s(“to”,?Attrs),??
- ???????
- ????%%将To转换成一个标准的JID:??
- ????%%#jid{user,resource,luser,lserver,lresource},具体参见jlib.erl中的定义??
- ????ToJID?=?case?To?of??
- ????????””?->??
- ????????????jlib:make_jid(User,?Server,?””);??
- ????????_?->??
- ????????????jlib:string_to_jid(To)??
- ????????end,??
- ??
- ????%%这里的El里的Attrs应该是一个元组组成的列表[{key1,value1},{key2,value2},{key3,value3},{key4,value4}…..]??
- ????%%下面的语句会从Attrs中删除key?为xmlns的元组,并返回新的El存放到NewEl1中??
- ????NewEl1?=?jlib:remove_attr(“xmlns”,?El),??
- ????NewEl?=?case?xml:get_attr_s(“xml:lang”,?Attrs)?of??
- ????????””?->??
- ????????????case?StateData#state.lang?of??
- ????????????””?->?NewEl1;??
- ????????????Lang?->??
- ????????????????xml:replace_tag_attr(“xml:lang”,?Lang,?NewEl1)??
- ????????????end;??
- ????????_?->??
- ????????????NewEl1??
- ????????end,??
- ??
- ????%%这里根据消息类型进行不同的处理??
- ????NewState?=??
- ????case?ToJID?of??
- ????????error?->??
- ????????case?xml:get_attr_s(“type”,?Attrs)?of??
- ????????????”error”?->?StateData;??
- ????????????”result”?->?StateData;??
- ????????????_?->??
- ????????????Err?=?jlib:make_error_reply(NewEl,??ERR_JID_MALFORMED),??
- ????????????send_element(StateData,?Err),??
- ????????????StateData??
- ????????end;??
- ????????_?->??
- ????????case?Name?of??
- ????????????”presence”?->??
- ????????????????????????%%如果为一个presence?消息,使用函数回调c2s_updatepresence参见2??
- ????????????????PresenceEl?=?ejabberd_hooks:run_fold(??
- ???????????????????????c2s_update_presence,??
- ???????????????????????Server,??
- ???????????????????????NewEl,??
- ???????????????????????[User,?Server]),??
- ?????????????????
- ??????????????????????????%%将调用结果发送回客户端??
- ??????????????????????????ejabberd_hooks:run(??
- ??????????????user_send_packet,??
- ??????????????Server,??
- ??????????????[FromJID,?ToJID,?PresenceEl]),??
- ????????????case?ToJID?of??
- ????????????????#jid{user?=?User,??
- ?????????????????server?=?Server,??
- ?????????????????resource?=?””}?->??
- ?????????????????DEBUG(“presence_update(~p,~n\t~p,~n\t~p)”,??
- ???????????????????????[FromJID,?PresenceEl,?StateData]),??
- ????????????????presence_update(FromJID,??
- ????????????????????????StateData);??
- ????????????????_?->??
- ????????????????presence_track(FromJID,??
- ???????????????????????????StateData)??
- ????????????end;??
- ????????????”iq”?->??
- ??????????????????????????%%iq?消息的处理.注册,添加好友等都是通过iq消息来发送的??
- ??????????????????????????case?jlib:iq_query_info(NewEl)?of??
- ????????????????#iq{xmlns?=?Xmlns}?=?IQ??
- ????????????????when?Xmlns?==??NS_PRIVACY;??
- ?????????????????Xmlns?==??NS_BLOCKING?->??
- ????????????????process_privacy_iq(??
- ??????????????????FromJID,?IQ,?StateData);??
- ????????????????_?->??
- ????????????????ejabberd_hooks:run(??
- ??????????????????user_send_packet,??
- ??????????????????Server,??
- ??????????????????[FromJID,?NewEl]),??
- ????????????????check_privacy_route(FromJID,?StateData,?FromJID,?NewEl),??
- ????????????????StateData??
- ????????????end;??
- ????????????”message”?->??
- ????????????????????????io:format(“message?arrvied~n”,[]),??
- ????????????ejabberd_hooks:run(user_send_packet,??
- ???????????????????????[FromJID,??
- ????????????check_privacy_route(FromJID,??
- ????????????????????????ToJID,??
- ????????????StateData;??
- ????????????_?->??
- ????????????StateData??
- ????????end??
- ????end,??
- ????ejabberd_hooks:run(c2s_loop_debug,?[{xmlstreamelement,?El}]),??
- ????fsm_next_state(session_established,?NewState).??
#p#副标题#e##p#分页标题#e#
2.下面是比较关键的ejabberd中函数调用的分析:
- ?????????????run_fold(Hook,?Host,?Val,?Args)?->??
- ??????????????????case?ets:lookup(hooks,?{Hook,?Host})?of??
- ?????????????????????????[{_,?Ls}]?->run_fold1(Ls,?Hook,?Args);??
- ?????????????????????????[]?->Val??
- ??????????????????end.??
- e=”code”?class=”plain”>PresenceEl?=?ejabberd_hooks:run_fold(??
- c2s_update_presence,??
- Server,??
- NewEl,??
- [User,?Server])??
在系统的ets库(内存中的一个数据库)中存有一个名为hooks的表,通过ets:lookup(hooks,{Hook,Host}) 可以找到一个 {_,Ls}的元组 (找不到就直接返回预定义的Val了),然后调用run_fold1(Ls,Hook,Val,Args).
Ls变量实际上是一个包含多个要具体调用的函数定义的列表,列表里面的元组分为两类:[{_Seq,Node,Moudle,Function} | Ls2]? [{_Seq,Module,Function} | Ls2],
run_fold1/4 的作用就是使用Args参数依次调用这个Ls列表里的方法.
run_fold1/4 最终会返回调用的结果出来.
所以从最终结果来看 ejabberd_hooks:run_fold/4? 方法就是去表hooks查找并调用所需的函数返回调用结果.
- PresenceEl?=?ejabberd_hooks:run_fold(??
- ???????????????????????c2s_update_presence,??
- ???????????????????????Server,??
- ???????????????????????NewEl,??
- ???????????????????????[User,?Server])??
针对上面的代码就是:
??????? 使用{c2s_update_presence,Server}作为key 在表hooks 中查找 要调用的方法列表,并使用[User,Server] 作为参数进行调用.
这个key具体找到什么样的方法呢? 我们可以在源码中查找下:
- root@ubuntu:??grep?*.erl?-e?c2s_update_presence??
查找结果中可以看到
mod_vcard_xupdate.erl?????????????? ejabberd_hooks:add(c2s_update_presence,Host
我们在mod_vcard_xupdate.erl中找到这段代码:
- start(Host,?_Opts)?->??
- ????mnesia:create_table(vcard_xupdate,??
- ????????????????????????[{disc_copies,?[node()]},??
- ?????????????????????????{attributes,?record_info(fields,?vcard_xupdate)}]),??
- ????ejabberd_hooks:add(c2s_update_presence,??
- ????????????????MODULE,?update_presence,?100),??
- ????ejabberd_hooks:add(vcard_set,?vcard_set,??
- ????ok.??
update_presence就所我们所要找的方法了.