I have the sample module as,
-module(mod_iq_test).
-behaviour(gen_mod).
-export([start/2,
init/2,
stop/1,
process_sm_iq/3
]).
-define(PROCNAME, ?MODULE).
-include("ejabberd.hrl").
-include("jlib.hrl").
-include("logger.hrl").
-define(NS_TEST, <<"http://jabber.org/protocol/test">>).
start(Host, Opts) ->
?INFO_MSG("Loading module 'mod_iqtest' v.01", []),
ok.
init(Host, _Opts) ->
IQDisc = gen_mod:get_opt(iqdisc, _Opts, fun(A) -> A end, one_queue),
gen_iq_handler:add_iq_handler(ejabberd_sm, Host, ?NS_TEST, ?MODULE, process_sm_iq, IQDisc),
?INFO_MSG("added iq handlers 'mod_iqtest' v.01", []),
ok.
stop(Host) ->
gen_iq_handler:remove_iq_handler(ejabberd_sm, Host, ?NS_TEST),
?INFO_MSG("Stoping module 'mod_iqtest' ", []),
ok.
process_sm_iq(_From, _To, IQ) ->
?INFO_MSG("Processing IQ Get query:~n ~p", [IQ]),
IQ#iq{type = result, sub_el = [{xmlel, <<"value">>, [], [{xmlcdata, <<"Hello World of Testing.">>}]}]}.
And while sending an IQ as,
<iq type='get'>
<query xmlns='http://jabber.org/protocol/test'/>
</iq>
<r xmlns="urn:xmpp:sm:3"/>
It throws an error as,
<iq type="error">
<query xmlns="http://jabber.org/protocol/test"/>
<error code="400" type="modify">
<bad-request xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"/>
<text xmlns="urn:ietf:params:xml:ns:xmpp-stanzas" xml:lang="en">Missing attribute 'id' in tag <iq/> qualified by namespace 'jabber:client'</text>
</error>
</iq>
Try providing an id attribute, as explained in the error response:
<iq type='get' id='aaa8821'>
<query xmlns='http://jabber.org/protocol/test'/>
</iq>