catch _:{xmpp_codec, Why} ->
Txt = xmpp:format_error(Why),
send_header(StateData, ?MYNAME, <<"1.0">>, ?MYLANG),
- send_element(StateData, xmpp:serr_not_well_formed(Txt, ?MYLANG)),
+ send_element(StateData, xmpp:serr_invalid_xml(Txt, ?MYLANG)),
{stop, normal, StateData}
end;
wait_for_stream(timeout, StateData) ->
wait_for_stream({xmlstreamstart, Name, Attrs}, StateData) ->
try xmpp:decode(#xmlel{name = Name, attrs = Attrs}) of
- #stream_start{xmlns = ?NS_COMPONENT, to = To} when is_record(To, jid) ->
+ #stream_start{xmlns = NS_COMPONENT, stream_xmlns = NS_STREAM}
+ when NS_COMPONENT /= ?NS_COMPONENT; NS_STREAM /= ?NS_STREAM ->
+ send_header(StateData, ?MYNAME),
+ send_element(StateData, xmpp:serr_invalid_namespace()),
+ {stop, normal, StateData};
+ #stream_start{to = To} when is_record(To, jid) ->
Host = To#jid.lserver,
send_header(StateData, Host),
HostOpts = case dict:is_key(Host, StateData#state.host_opts) of
end,
{next_state, wait_for_handshake,
StateData#state{host = Host, host_opts = HostOpts}};
- #stream_start{xmlns = ?NS_COMPONENT} ->
- send_header(StateData, ?MYNAME),
- send_element(StateData, xmpp:serr_improper_addressing()),
- {stop, normal, StateData};
#stream_start{} ->
send_header(StateData, ?MYNAME),
- send_element(StateData, xmpp:serr_invalid_namespace()),
+ send_element(StateData, xmpp:serr_improper_addressing()),
{stop, normal, StateData}
catch _:{xmpp_codec, Why} ->
Txt = xmpp:format_error(Why),
To = xmpp:get_to(El),
Lang = xmpp:get_lang(El),
if From == undefined orelse To == undefined ->
- send_error(StateData, El, xmpp:err_jid_malformed());
+ Txt = <<"Missing 'from' or 'to' attribute">>,
+ send_error(StateData, El, xmpp:err_jid_malformed(Txt, Lang));
true ->
FromJID = case StateData#state.check_from of
false ->
From;
_ ->
%% The default is the standard behaviour in XEP-0114
- case From of
- #jid{lserver = Server} ->
- case dict:is_key(Server, StateData#state.host_opts) of
- true -> From;
- false -> error
- end;
- _ -> error
+ Server = From#jid.lserver,
+ case dict:is_key(Server, StateData#state.host_opts) of
+ true -> From;
+ false -> error
end
end,
if FromJID /= error ->
ejabberd_router:route(FromJID, To, El);
true ->
- Txt = <<"Incorrect value of 'from' or 'to' attribute">>,
+ Txt = <<"Improper domain part of 'from' attribute">>,
send_error(StateData, El, xmpp:err_not_allowed(Txt, Lang))
end
end,
Err ->
{skip, {riak_not_available, Err}}
end;
+do_init_per_group(component, Config) ->
+ Server = ?config(server, Config),
+ Port = ?config(component_port, Config),
+ set_opt(xmlns, ?NS_COMPONENT,
+ set_opt(server, <<"component.", Server/binary>>,
+ set_opt(type, component,
+ set_opt(server_port, Port,
+ set_opt(stream_version, <<"">>,
+ set_opt(lang, <<"">>, Config))))));
do_init_per_group(_GroupName, Config) ->
Pid = start_event_relay(),
set_opt(event_relay, Pid, Config).
ok;
end_per_group(riak, _Config) ->
ok;
+end_per_group(component, _Config) ->
+ ok;
end_per_group(_GroupName, Config) ->
stop_event_relay(Config),
ok.
[{generic, [parallel],
[test_connect_bad_xml,
test_connect_unknown_ns,
- test_connect_bad_ns_client,
+ test_connect_bad_xmlns,
test_connect_bad_ns_stream,
test_connect_bad_lang,
test_connect_bad_to,
test_auth_fail,
test_unregister]}].
+component_tests() ->
+ [{component_tests, [sequence],
+ [test_connect_bad_xml,
+ test_connect_unknown_ns,
+ test_connect_bad_xmlns,
+ test_connect_bad_ns_stream,
+ test_connect_missing_to,
+ test_connect,
+ test_auth,
+ test_auth_fail,
+ component_missing_address,
+ component_invalid_from,
+ component_send,
+ bad_nonza,
+ codec_failure]}].
+
groups() ->
[{ldap, [sequence], ldap_tests()},
{extauth, [sequence], extauth_tests()},
{no_db, [sequence], no_db_tests()},
+ {component, [sequence], component_tests()},
{mnesia, [sequence], db_tests(mnesia)},
{redis, [sequence], db_tests(redis)},
{mysql, [sequence], db_tests(mysql)},
{riak, [sequence], db_tests(riak)}].
all() ->
- [%%{group, ldap},
+ [{group, component},
+ %%{group, ldap},
{group, no_db},
{group, mnesia},
%%{group, redis},
Config.
test_connect_bad_xml(Config) ->
- Config0 = init_stream(set_opt(ns_client, <<"'">>, Config)),
+ Config0 = init_stream(set_opt(xmlns, <<"'">>, Config)),
?recv1(#stream_error{reason = 'not-well-formed'}),
?recv1({xmlstreamend, <<"stream:stream">>}),
close_socket(Config0).
test_connect_unknown_ns(Config) ->
- Config0 = init_stream(set_opt(ns_client, <<"wrong">>, Config)),
- ?recv1(#stream_error{reason = 'not-well-formed'}),
+ Config0 = init_stream(set_opt(xmlns, <<"wrong">>, Config)),
+ ?recv1(#stream_error{reason = 'invalid-xml'}),
?recv1({xmlstreamend, <<"stream:stream">>}),
close_socket(Config0).
-test_connect_bad_ns_client(Config) ->
- Config0 = init_stream(set_opt(ns_client, ?NS_SERVER, Config)),
+test_connect_bad_xmlns(Config) ->
+ Config0 = init_stream(set_opt(xmlns, ?NS_SERVER, Config)),
?recv1(#stream_error{reason = 'invalid-namespace'}),
?recv1({xmlstreamend, <<"stream:stream">>}),
close_socket(Config0).
test_connect(Config) ->
disconnect(connect(Config)).
+test_component_connect(Config) ->
+ disconnect(component_connect(Config)).
+
+component_connect(Config) ->
+ init_stream(Config).
+
test_starttls(Config) ->
case ?config(starttls, Config) of
true ->
?recv1({xmlstreamend, <<"stream:stream">>}),
close_socket(Config).
+component_missing_address(Config) ->
+ Server = server_jid(Config),
+ #iq{type = error} = send_recv(Config, #iq{type = get, from = Server}),
+ #iq{type = error} = send_recv(Config, #iq{type = get, to = Server}),
+ disconnect(Config).
+
+component_invalid_from(Config) ->
+ From = jid:make(randoms:get_string()),
+ To = jid:make(randoms:get_string()),
+ #iq{type = error} =
+ send_recv(Config, #iq{type = get, from = From, to = To}),
+ disconnect(Config).
+
+component_send(Config) ->
+ JID = my_jid(Config),
+ send(Config, #message{from = JID, to = JID}),
+ #message{from = JID, to = JID} = recv(),
+ disconnect(Config).
+
auth_md5(Config) ->
Mechs = ?config(mechs, Config),
case lists:member(<<"DIGEST-MD5">>, Mechs) of
disconnect(auth(Config)).
test_auth_fail(Config0) ->
- Config = set_opt(user, <<"wrong">>, Config0),
+ Config = set_opt(user, <<"wrong">>,
+ set_opt(password, <<"wrong">>, Config0)),
disconnect(auth(Config, _ShouldFail = true)).
test_bind(Config) ->
disconnect(Config).
codec_failure(Config) ->
- #iq{type = error} = send_recv(Config, #iq{type = wrong}),
+ JID = my_jid(Config),
+ #iq{type = error} =
+ send_recv(Config, #iq{type = wrong, from = JID, to = JID}),
disconnect(Config).
unsupported_query(Config) ->
<<>>)},
Storage = #bookmark_storage{conference = [Conference]},
StorageXMLOut = xmpp_codec:encode(Storage),
+ WrongEl = #xmlel{name = <<"wrong">>},
#iq{type = error} =
- send_recv(Config, #iq{type = get, sub_els = [#private{}],
- to = server_jid(Config)}),
+ send_recv(Config, #iq{type = get,
+ sub_els = [#private{xml_els = [WrongEl]}]}),
#iq{type = result, sub_els = []} =
send_recv(
Config, #iq{type = set,
- sub_els = [#private{xml_els = [StorageXMLOut]}]}),
+ sub_els = [#private{xml_els = [WrongEl, StorageXMLOut]}]}),
#iq{type = result,
sub_els = [#private{xml_els = [StorageXMLIn]}]} =
send_recv(
port: @@web_port@@
module: ejabberd_http
captcha: true
+ -
+ port: @@component_port@@
+ module: ejabberd_service
+ password: >-
+ @@password@@
loglevel: @@loglevel@@
max_fsm_queue: 1000
modules:
{ok, CWD} = file:get_cwd(),
{ok, _} = file:copy(CertFile, filename:join([CWD, "cert.pem"])),
{ok, CfgContentTpl} = file:read_file(ConfigPathTpl),
+ Password = <<"password!@#$%^&*()'\"`~<>+-/;:_=[]{}|\\">>,
CfgContent = process_config_tpl(CfgContentTpl, [
{c2s_port, 5222},
{loglevel, 5},
{s2s_port, 5269},
+ {component_port, 5270},
{web_port, 5280},
+ {password, Password},
{mysql_server, <<"localhost">>},
{mysql_port, 3306},
{mysql_db, <<"ejabberd_test">>},
application:set_env(mnesia, dir, MnesiaDir),
[{server_port, ct:get_config(c2s_port, 5222)},
{server_host, "localhost"},
+ {component_port, ct:get_config(component_port, 5270)},
{server, ?COMMON_VHOST},
{user, <<"test_single!#$%^*()`~+-;_=[]{}|\\">>},
{master_nick, <<"master_nick!@#$%^&*()'\"`~<>+-/;:_=[]{}|\\">>},
{slave_nick, <<"slave_nick!@#$%^&*()'\"`~<>+-/;:_=[]{}|\\">>},
{room_subject, <<"hello, world!@#$%^&*()'\"`~<>+-/;:_=[]{}|\\">>},
{certfile, CertFile},
- {ns_client, ?NS_CLIENT},
+ {type, client},
+ {xmlns, ?NS_CLIENT},
{ns_stream, ?NS_STREAM},
{stream_version, <<"1.0">>},
{stream_id, <<"">>},
{resource, <<"resource!@#$%^&*()'\"`~<>+-/;:_=[]{}|\\">>},
{master_resource, <<"master_resource!@#$%^&*()'\"`~<>+-/;:_=[]{}|\\">>},
{slave_resource, <<"slave_resource!@#$%^&*()'\"`~<>+-/;:_=[]{}|\\">>},
- {password, <<"password!@#$%^&*()'\"`~<>+-/;:_=[]{}|\\">>},
+ {password, Password},
{backends, get_config_backends()}
|Config].
stream_header(Config) ->
NSStream = ?config(ns_stream, Config),
- NSClient = ?config(ns_client, Config),
- Lang = ?config(lang, Config),
+ XMLNS = ?config(xmlns, Config),
+ Lang = case ?config(lang, Config) of
+ <<"">> -> <<"">>;
+ L -> iolist_to_binary(["xml:lang='", L, "'"])
+ end,
To = case ?config(server, Config) of
<<"">> -> <<"">>;
Server -> <<"to='", Server/binary, "'">>
end,
- Version = ?config(stream_version, Config),
+ Version = case ?config(stream_version, Config) of
+ <<"">> -> <<"">>;
+ V -> <<"version='", V/binary, "'">>
+ end,
io_lib:format("<?xml version='1.0'?><stream:stream "
- "xmlns:stream='~s' xmlns='~s' ~s "
- "version='~s' xml:lang='~s'>",
- [NSStream, NSClient, To, Version, Lang]).
+ "xmlns:stream='~s' xmlns='~s' ~s ~s ~s>",
+ [NSStream, XMLNS, To, Version, Lang]).
connect(Config) ->
- process_stream_features(init_stream(Config)).
+ NewConfig = init_stream(Config),
+ case ?config(type, NewConfig) of
+ client -> process_stream_features(NewConfig);
+ component -> NewConfig
+ end.
init_stream(Config) ->
Version = ?config(stream_version, Config),
[binary, {packet, 0}, {active, false}]),
NewConfig = set_opt(socket, Sock, Config),
ok = send_text(NewConfig, stream_header(NewConfig)),
- #stream_start{id = ID, xmlns = ?NS_CLIENT,
- version = Version} = recv(),
+ XMLNS = case ?config(type, Config) of
+ client -> ?NS_CLIENT;
+ component -> ?NS_COMPONENT
+ end,
+ #stream_start{id = ID, xmlns = XMLNS, version = Version} = recv(),
set_opt(stream_id, ID, NewConfig).
process_stream_features(Config) ->
disconnect(Config) ->
Socket = ?config(socket, Config),
- ok = ejabberd_socket:send(Socket, ?STREAM_TRAILER),
+ try
+ ok = send_text(Config, ?STREAM_TRAILER)
+ catch exit:normal ->
+ ok
+ end,
{xmlstreamend, <<"stream:stream">>} = recv(),
ejabberd_socket:close(Socket),
Config.
auth(Config, false).
auth(Config, ShouldFail) ->
+ Type = ?config(type, Config),
Mechs = ?config(mechs, Config),
HaveMD5 = lists:member(<<"DIGEST-MD5">>, Mechs),
HavePLAIN = lists:member(<<"PLAIN">>, Mechs),
auth_SASL(<<"PLAIN">>, Config, ShouldFail);
HaveMD5 ->
auth_SASL(<<"DIGEST-MD5">>, Config, ShouldFail);
- true ->
- auth_legacy(Config, false, ShouldFail)
+ Type == client ->
+ auth_legacy(Config, false, ShouldFail);
+ Type == component ->
+ auth_component(Config, ShouldFail)
end.
bind(Config) ->
- #iq{type = result, sub_els = [#bind{}]} =
- send_recv(
- Config,
- #iq{type = set,
- sub_els = [#bind{resource = ?config(resource, Config)}]}),
+ case ?config(type, Config) of
+ client ->
+ #iq{type = result, sub_els = [#bind{}]} =
+ send_recv(
+ Config,
+ #iq{type = set,
+ sub_els = [#bind{resource = ?config(resource, Config)}]});
+ component ->
+ ok
+ end,
Config.
open_session(Config) ->
end
end.
+auth_component(Config, ShouldFail) ->
+ StreamID = ?config(stream_id, Config),
+ Password = ?config(password, Config),
+ Digest = p1_sha:sha(<<StreamID/binary, Password/binary>>),
+ send(Config, #handshake{data = Digest}),
+ case recv() of
+ #handshake{} when ShouldFail ->
+ ct:fail(component_auth_should_have_failed);
+ #handshake{} ->
+ Config;
+ #stream_error{reason = 'not-authorized'} when ShouldFail ->
+ Config;
+ #stream_error{reason = 'not-authorized'} ->
+ ct:fail(component_auth_failed)
+ end.
+
auth_SASL(Mech, Config) ->
auth_SASL(Mech, Config, false).