aboutsummaryrefslogtreecommitdiff
path: root/doc/dev.html
diff options
context:
space:
mode:
authorAlexey Shchepin <alexey@process-one.net>2003-11-11 20:51:04 +0000
committerAlexey Shchepin <alexey@process-one.net>2003-11-11 20:51:04 +0000
commitc248bffd54ed08034f22e98e13b9612f685592d4 (patch)
tree59f9731a2dca9d1fd328abec4642671e07717f36 /doc/dev.html
parent* src/ejabberd.cfg.example: Updated (diff)
* doc/dev.tex: Developers documentation (not completed)
* src/ejabberd_c2s.erl: Better handling of malformed JIDs * src/mod_register.erl (try_register/2): Now returns "jid malformed" error if user name is invalid SVN Revision: 174
Diffstat (limited to '')
-rw-r--r--doc/dev.html364
1 files changed, 364 insertions, 0 deletions
diff --git a/doc/dev.html b/doc/dev.html
new file mode 100644
index 000000000..ad12f39dd
--- /dev/null
+++ b/doc/dev.html
@@ -0,0 +1,364 @@
+<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN"
+ "http://www.w3.org/TR/REC-html40/loose.dtd">
+<HTML>
+<HEAD><TITLE>Ejabberd Developers Guide</TITLE>
+
+<META http-equiv="Content-Type" content="text/html; charset=ISO8859-1">
+<META name="GENERATOR" content="hevea 1.06">
+</HEAD>
+<BODY >
+<!--HEVEA command line is: /usr/bin/hevea -charset ISO8859-1 dev.tex -->
+<!--HTMLHEAD-->
+<!--ENDHTML-->
+<!--PREFIX <ARG ></ARG>-->
+<!--CUT DEF section 1 -->
+
+
+
+<H1 ALIGN=center>Ejabberd Developers Guide</H1>
+
+<H3 ALIGN=center>Alexey Shchepin<BR>
+<A HREF="mailto:alexey@sevcom.net"><TT>mailto:alexey@sevcom.net</TT></A><BR>
+<A HREF="xmpp:aleksey@jabber.ru"><TT>xmpp:aleksey@jabber.ru</TT></A></H3>
+
+<H3 ALIGN=center>September 10, 2003</H3><DIV ALIGN=center>
+
+ <IMG SRC="logo.png">
+
+
+ </DIV><BR>
+<BR>
+
+
+<!--TOC section Table of Contents-->
+
+<H2>Table of Contents</H2><!--SEC END -->
+
+<UL><LI>
+<A HREF="#htoc1">1&nbsp;&nbsp;Introduction</A>
+<UL><LI>
+<A HREF="#htoc2">1.1&nbsp;&nbsp;How it works</A>
+<UL><LI>
+<A HREF="#htoc3">1.1.1&nbsp;&nbsp;Router</A>
+<LI><A HREF="#htoc4">1.1.2&nbsp;&nbsp;Local Router</A>
+<LI><A HREF="#htoc5">1.1.3&nbsp;&nbsp;Session Manager</A>
+<LI><A HREF="#htoc6">1.1.4&nbsp;&nbsp;S2S Manager</A>
+</UL>
+</UL>
+<LI><A HREF="#htoc7">2&nbsp;&nbsp;XML representation</A>
+<LI><A HREF="#htoc8">3&nbsp;&nbsp;Module <TT>xml</TT></A>
+<LI><A HREF="#htoc9">4&nbsp;&nbsp;<TT>ejabberd</TT> modules</A>
+<UL><LI>
+<A HREF="#htoc10">4.1&nbsp;&nbsp;<CODE>gen_mod</CODE> behaviour</A>
+<LI><A HREF="#htoc11">4.2&nbsp;&nbsp;Module <CODE>gen_iq_handler</CODE></A>
+<LI><A HREF="#htoc12">4.3&nbsp;&nbsp;Services</A>
+</UL>
+</UL>
+
+<!--TOC section Introduction-->
+
+<H2><A NAME="htoc1">1</A>&nbsp;&nbsp;Introduction</H2><!--SEC END -->
+
+<A NAME="sec:intro"></A>
+<TT>ejabberd</TT> is a Free and Open Source fault-tolerant distributed Jabber
+server. It is writen mostly in Erlang.<BR>
+<BR>
+The main features of <TT>ejabberd</TT> is:
+<UL><LI>
+Works on most of popular platforms: *nix (tested on Linux and FreeBSD)
+ and Win32
+<LI>Distributed: You can run <TT>ejabberd</TT> on a cluster of machines and all of
+ them will serve one Jabber domain.
+<LI>Fault-tolerance: You can setup an <TT>ejabberd</TT> cluster so that all the
+ information required for a properly working service will be stored
+ permanently on more than one node. This means that if one of the nodes
+ crashes, then the others will continue working without disruption.
+ You can also add or replace more nodes ``on the fly''.
+<LI>Built-in <A HREF="http://www.jabber.org/jeps/jep-0045.html">Multi-User
+ Chat</A> service
+<LI>Built-in IRC transport
+<LI>Built-in
+ <A HREF="http://www.jabber.org/jeps/jep-0060.html">Publish-Subscribe</A>
+ service
+<LI>Built-in Jabber Users Directory service based on users vCards
+<LI>Support for
+ <A HREF="http://www.jabber.org/jeps/jep-0030.html">JEP-0030</A>
+ (Service Discovery).
+<LI>Support for
+ <A HREF="http://www.jabber.org/jeps/jep-0039.html">JEP-0039</A>
+ (Statistics Gathering).
+<LI>Support for <TT>xml:lang</TT> attribute in many XML elements
+</UL>
+<!--TOC subsection How it works-->
+
+<H3><A NAME="htoc2">1.1</A>&nbsp;&nbsp;How it works</H3><!--SEC END -->
+
+<A NAME="sec:howitworks"></A>
+A Jabber domain is served by one or more <TT>ejabberd</TT> nodes. These nodes can
+be run on different machines that are connected via a network. They all must
+have the ability to connect to port 4369 of all another nodes, and must have
+the same magic cookie (see Erlang/OTP documentation, in other words the file
+<TT>~ejabberd/.erlang.cookie</TT> must be the same on all nodes). This is
+needed because all nodes exchange information about connected users, S2S
+connections, registered services, etc...<BR>
+<BR>
+Each <TT>ejabberd</TT> node have following modules:
+<UL><LI>
+router;
+<LI>local router.
+<LI>session manager;
+<LI>S2S manager;
+</UL>
+<!--TOC subsubsection Router-->
+
+<H4><A NAME="htoc3">1.1.1</A>&nbsp;&nbsp;Router</H4><!--SEC END -->
+
+This module is the main router of Jabber packets on each node. It routes
+them based on their destinations domains. It has two tables: local and global
+routes. First, domain of packet destination searched in local table, and if it
+found, then the packet is routed to appropriate process. If no, then it
+searches in global table, and is routed to the appropriate <TT>ejabberd</TT> node or
+process. If it does not exists in either tables, then it sent to the S2S
+manager.<BR>
+<BR>
+<!--TOC subsubsection Local Router-->
+
+<H4><A NAME="htoc4">1.1.2</A>&nbsp;&nbsp;Local Router</H4><!--SEC END -->
+
+This module routes packets which have a destination domain equal to this server
+name. If destination JID has a non-empty user part, then it routed to the
+session manager, else it is processed depending on it's content.<BR>
+<BR>
+<!--TOC subsubsection Session Manager-->
+
+<H4><A NAME="htoc5">1.1.3</A>&nbsp;&nbsp;Session Manager</H4><!--SEC END -->
+
+This module routes packets to local users. It searches for what user resource
+packet must be sended via presence table. If this resource is connected to
+this node, it is routed to C2S process, if it connected via another node, then
+the packet is sent to session manager on that node.<BR>
+<BR>
+<!--TOC subsubsection S2S Manager-->
+
+<H4><A NAME="htoc6">1.1.4</A>&nbsp;&nbsp;S2S Manager</H4><!--SEC END -->
+
+This module routes packets to other Jabber servers. First, it checks if an
+open S2S connection from the domain of the packet source to the domain of
+packet destination already exists. If it is open on another node, then it
+routes the packet to S2S manager on that node, if it is open on this node, then
+it is routed to the process that serves this connection, and if a connection
+does not exist, then it is opened and registered.<BR>
+<BR>
+<!--TOC section XML representation-->
+
+<H2><A NAME="htoc7">2</A>&nbsp;&nbsp;XML representation</H2><!--SEC END -->
+
+<A NAME="sec:xmlrepr"></A>
+Each XML stanza represented as following tuple:
+<PRE>
+XMLElement = {xmlelement, Name, Attrs, [ElementOrCDATA]}
+ Name = string()
+ Attrs = [Attr]
+ Attr = {Key, Val}
+ Key = string()
+ Val = string()
+ ElementOrCDATA = XMLElement | CDATA
+ CDATA = {xmlcdata, string()}
+</PRE>E.&nbsp;g. this stanza:
+<PRE>
+&lt;message to='test@conference.e.localhost' type='groupchat'&gt;
+ &lt;body&gt;test&lt;/body&gt;
+&lt;/message&gt;
+</PRE>represented as following structure:
+<PRE>
+{xmlelement, "message",
+ [{"to", "test@conference.e.localhost"},
+ {"type", "groupchat"}],
+ [{xmlelement, "body",
+ [],
+ [{xmlcdata, "test"}]}]}}
+</PRE>
+<!--TOC section Module <TT>xml</TT>-->
+
+<H2><A NAME="htoc8">3</A>&nbsp;&nbsp;Module <TT>xml</TT></H2><!--SEC END -->
+
+<A NAME="sec:xmlmod"></A>
+<DL COMPACT=compact><DT>
+<CODE><B>element_to_string(El) -&gt; string()</B></CODE><DD>
+<PRE>
+El = XMLElement
+</PRE>Returns string representation of XML stanza <TT>El</TT>.<BR>
+<BR>
+<DT><CODE><B>crypt(S) -&gt; string()</B></CODE><DD>
+<PRE>
+S = string()
+</PRE>Returns string which correspond to <TT>S</TT> with encoded XML special
+ characters.<BR>
+<BR>
+<DT><CODE><B>remove_cdata(ECList) -&gt; EList</B></CODE><DD>
+<PRE>
+ECList = [ElementOrCDATA]
+EList = [XMLElement]
+</PRE><TT>EList</TT> is a list of all non-CDATA elements of ECList.<BR>
+<BR>
+<DT><CODE><B>get_path_s(El, Path) -&gt; Res</B></CODE><DD>
+<PRE>
+El = XMLElement
+Path = [PathItem]
+PathItem = PathElem | PathAttr | PathCDATA
+PathElem = {elem, Name}
+PathAttr = {attr, Name}
+PathCDATA = cdata
+Name = string()
+Res = string() | XMLElement
+</PRE>If <TT>Path</TT> is empty, then returns <TT>El</TT>. Else sequentially
+ consider elements of <TT>Path</TT>. Each element is one of:
+ <DL COMPACT=compact><DT>
+ <CODE><B>{elem, Name}</B></CODE><DD> <TT>Name</TT> is name of subelement of
+ <TT>El</TT>, if such element exists, then this element considered in
+ following steps, else returns empty string.
+ <DT><CODE><B>{attr, Name}</B></CODE><DD> If <TT>El</TT> have attribute <TT>Name</TT>, then
+ returns value of this attribute, else returns empty string.
+ <DT><CODE><B>cdata</B></CODE><DD> Returns CDATA of <TT>El</TT>.
+ </DL><BR>
+<BR>
+<DT><B>TODO:</B><DD>
+<PRE>
+ get_cdata/1, get_tag_cdata/1
+ get_attr/2, get_attr_s/2
+ get_tag_attr/2, get_tag_attr_s/2
+ get_subtag/2
+</PRE></DL>
+<!--TOC section <TT>ejabberd</TT> modules-->
+
+<H2><A NAME="htoc9">4</A>&nbsp;&nbsp;<TT>ejabberd</TT> modules</H2><!--SEC END -->
+
+<A NAME="sec:emods"></A>
+<!--TOC subsection <CODE>gen_mod</CODE> behaviour-->
+
+<H3><A NAME="htoc10">4.1</A>&nbsp;&nbsp;<CODE>gen_mod</CODE> behaviour</H3><!--SEC END -->
+
+<A NAME="sec:genmod"></A>
+TBD<BR>
+<BR>
+<!--TOC subsection Module <CODE>gen_iq_handler</CODE>-->
+
+<H3><A NAME="htoc11">4.2</A>&nbsp;&nbsp;Module <CODE>gen_iq_handler</CODE></H3><!--SEC END -->
+
+<A NAME="sec:geniqhandl"></A>
+The module <CODE>gen_iq_handler</CODE> allows to easily write handlers for IQ packets
+of particular XML namespaces that addressed to server or to users bare JIDs.<BR>
+<BR>
+In this module the following functions are defined:
+<DL COMPACT=compact><DT>
+<CODE><B>add_iq_handler(Component, NS, Module, Function, Type)</B></CODE><DD>
+<PRE>
+Component = Module = Function = atom()
+NS = string()
+Type = no_queue | one_queue | parallel
+</PRE>Registers function <CODE>Module:Function</CODE> as handler for IQ packets that
+ contain child of namespace <CODE>NS</CODE> in <CODE>Component</CODE>. Queueing
+ discipline is <CODE>Type</CODE>. There are at least two components defined:
+ <DL COMPACT=compact><DT>
+ <CODE><B>ejabberd_local</B></CODE><DD> Handles packets that addressed to server JID;
+ <DT><CODE><B>ejabberd_sm</B></CODE><DD> Handles packets that addressed to users bare JIDs.
+ </DL>
+<DT><CODE><B>remove_iq_handler(Component, NS)</B></CODE><DD>
+<PRE>
+Component = atom()
+NS = string()
+</PRE>Removes IQ handler for namespace <CODE>NS</CODE> from <CODE>Component</CODE>.
+</DL>
+Handler function must have the following type:
+<DL COMPACT=compact><DT>
+<CODE><B>Module:Function(From, To, IQ)</B></CODE><DD>
+<PRE>
+From = To = jid()
+</PRE></DL>
+<PRE>
+-module(mod_cputime).
+
+-behaviour(gen_mod).
+
+-export([start/1,
+ stop/0,
+ process_local_iq/3]).
+
+-include("ejabberd.hrl").
+-include("jlib.hrl").
+
+-define(NS_CPUTIME, "ejabberd:cputime").
+
+start(Opts) -&gt;
+ IQDisc = gen_mod:get_opt(iqdisc, Opts, one_queue),
+ gen_iq_handler:add_iq_handler(ejabberd_local, ?NS_CPUTIME,
+ ?MODULE, process_local_iq, IQDisc).
+
+stop() -&gt;
+ gen_iq_handler:remove_iq_handler(ejabberd_local, ?NS_CPUTIME).
+
+process_local_iq(From, To, {iq, ID, Type, XMLNS, SubEl}) -&gt;
+ case Type of
+ set -&gt;
+ {iq, ID, error, XMLNS,
+ [SubEl, ?ERR_NOT_ALLOWED]};
+ get -&gt;
+ CPUTime = element(1, erlang:statistics(runtime))/1000,
+ SCPUTime = lists:flatten(io_lib:format("~.3f", CPUTime)),
+ {iq, ID, result, XMLNS,
+ [{xmlelement, "query",
+ [{"xmlns", ?NS_CPUTIME}],
+ [{xmlelement, "cputime", [], [{xmlcdata, SCPUTime}]}]}]}
+ end.
+</PRE>
+<!--TOC subsection Services-->
+
+<H3><A NAME="htoc12">4.3</A>&nbsp;&nbsp;Services</H3><!--SEC END -->
+
+<A NAME="sec:services"></A>
+TBD<BR>
+<BR>
+TODO: use <CODE>proc_lib</CODE>
+<PRE>
+-module(mod_echo).
+
+-behaviour(gen_mod).
+
+-export([start/1, init/1, stop/0]).
+
+-include("ejabberd.hrl").
+-include("jlib.hrl").
+
+start(Opts) -&gt;
+ Host = gen_mod:get_opt(host, Opts, "echo." ++ ?MYNAME),
+ register(ejabberd_mod_echo, spawn(?MODULE, init, [Host])).
+
+init(Host) -&gt;
+ ejabberd_router:register_local_route(Host),
+ loop(Host).
+
+loop(Host) -&gt;
+ receive
+ {route, From, To, Packet} -&gt;
+ ejabberd_router:route(To, From, Packet),
+ loop(Host);
+ stop -&gt;
+ ejabberd_router:unregister_local_route(Host),
+ ok;
+ _ -&gt;
+ loop(Host)
+ end.
+
+stop() -&gt;
+ ejabberd_mod_echo ! stop,
+ ok.
+</PRE>
+<!--HTMLFOOT-->
+<!--ENDHTML-->
+<!--FOOTER-->
+<HR SIZE=2>
+<BLOCKQUOTE><EM>This document was translated from L<sup>A</sup>T<sub>E</sub>X by
+</EM><A HREF="http://pauillac.inria.fr/~maranget/hevea/index.html"><EM>H<FONT SIZE=2><sup>E</sup></FONT>V<FONT SIZE=2><sup>E</sup></FONT>A</EM></A><EM>.
+</EM></BLOCKQUOTE>
+</BODY>
+</HTML>