1.客户端连接服务器,并发送消息给服务器的处理在ejabberd_c2s中。
ejabberd_c2s 是一个gen_fsm 状态机,在 ejabberd启动时装载。
初始状态为 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,server,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, PresenceEl, StateData); _ -> presence_track(FromJID, ToJID, PresenceEl, 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, ToJID, IQ, StateData); _ -> ejabberd_hooks:run( user_send_packet, Server, [FromJID, ToJID, NewEl]), check_privacy_route(FromJID, StateData, FromJID, ToJID, NewEl), StateData end; "message" -> io:format("message arrvied~n",[]), ejabberd_hooks:run(user_send_packet, Server, [FromJID, ToJID, NewEl]), check_privacy_route(FromJID, StateData, FromJID, ToJID, NewEl), StateData; _ -> StateData end end, ejabberd_hooks:run(c2s_loop_debug, [{xmlstreamelement, El}]), fsm_next_state(session_established, NewState).
2.下面是比较关键的ejabberd中函数调用的分析:
run_fold(Hook, Host, Val, Args) -> case ets:lookup(hooks, {Hook, Host}) of [{_, Ls}] ->run_fold1(Ls, Hook, Val, Args); [] ->Val end. <pre name="code" class="plain">PresenceEl = ejabberd_hooks:run_fold( c2s_update_presence, Server, NewEl, [User, Server])</pre>
在系统的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, Host, ?MODULE, update_presence, 100), ejabberd_hooks:add(vcard_set, Host, ?MODULE, vcard_set, 100), ok.
update_presence就所我们所要找的方法了.