26
1
mirror of https://github.com/processone/ejabberd.git synced 2025-01-03 18:02:28 +01:00

Merge pull request #1646 from weiss/custom-headers

Add custom_headers option for ejabberd_http listeners
This commit is contained in:
Christophe Romain 2017-03-30 12:47:42 +02:00 committed by GitHub
commit 997ac58329
3 changed files with 37 additions and 17 deletions

View File

@ -68,6 +68,7 @@
end_of_request = false, end_of_request = false,
options = [], options = [],
default_host, default_host,
custom_headers,
trail = <<>>, trail = <<>>,
addr_re addr_re
}). }).
@ -167,10 +168,15 @@ init({SockMod, Socket}, Opts) ->
DefaultHost = gen_mod:get_opt(default_host, Opts, fun(A) -> A end, undefined), DefaultHost = gen_mod:get_opt(default_host, Opts, fun(A) -> A end, undefined),
{ok, RE} = re:compile(<<"^(?:\\[(.*?)\\]|(.*?))(?::(\\d+))?$">>), {ok, RE} = re:compile(<<"^(?:\\[(.*?)\\]|(.*?))(?::(\\d+))?$">>),
CustomHeaders = gen_mod:get_opt(custom_headers, Opts,
fun expand_custom_headers/1,
[]),
?INFO_MSG("started: ~p", [{SockMod1, Socket1}]), ?INFO_MSG("started: ~p", [{SockMod1, Socket1}]),
State = #state{sockmod = SockMod1, State = #state{sockmod = SockMod1,
socket = Socket1, socket = Socket1,
default_host = DefaultHost, default_host = DefaultHost,
custom_headers = CustomHeaders,
options = Opts, options = Opts,
request_handlers = RequestHandlers, request_handlers = RequestHandlers,
addr_re = RE}, addr_re = RE},
@ -309,6 +315,7 @@ process_header(State, Data) ->
trail = State3#state.trail, trail = State3#state.trail,
options = State#state.options, options = State#state.options,
default_host = State#state.default_host, default_host = State#state.default_host,
custom_headers = State#state.custom_headers,
request_handlers = State#state.request_handlers, request_handlers = State#state.request_handlers,
addr_re = State#state.addr_re}; addr_re = State#state.addr_re};
_ -> _ ->
@ -316,6 +323,7 @@ process_header(State, Data) ->
trail = State3#state.trail, trail = State3#state.trail,
options = State#state.options, options = State#state.options,
default_host = State#state.default_host, default_host = State#state.default_host,
custom_headers = State#state.custom_headers,
request_handlers = State#state.request_handlers, request_handlers = State#state.request_handlers,
addr_re = State#state.addr_re} addr_re = State#state.addr_re}
end; end;
@ -323,6 +331,7 @@ process_header(State, Data) ->
#state{end_of_request = true, #state{end_of_request = true,
options = State#state.options, options = State#state.options,
default_host = State#state.default_host, default_host = State#state.default_host,
custom_headers = State#state.custom_headers,
request_handlers = State#state.request_handlers, request_handlers = State#state.request_handlers,
addr_re = State#state.addr_re} addr_re = State#state.addr_re}
end. end.
@ -446,6 +455,7 @@ process_request(#state{request_method = Method,
request_tp = TP, request_tp = TP,
request_headers = RequestHeaders, request_headers = RequestHeaders,
request_handlers = RequestHandlers, request_handlers = RequestHandlers,
custom_headers = CustomHeaders,
trail = Trail} = State) -> trail = Trail} = State) ->
case extract_path_query(State) of case extract_path_query(State) of
{State2, false} -> {State2, false} ->
@ -478,18 +488,21 @@ process_request(#state{request_method = Method,
ip = IP}, ip = IP},
Res = case process(RequestHandlers, Request, Socket, SockMod, Trail) of Res = case process(RequestHandlers, Request, Socket, SockMod, Trail) of
El when is_record(El, xmlel) -> El when is_record(El, xmlel) ->
make_xhtml_output(State, 200, [], El); make_xhtml_output(State, 200, CustomHeaders, El);
{Status, Headers, El} {Status, Headers, El}
when is_record(El, xmlel) -> when is_record(El, xmlel) ->
make_xhtml_output(State, Status, Headers, El); make_xhtml_output(State, Status,
Headers ++ CustomHeaders, El);
Output when is_binary(Output) or is_list(Output) -> Output when is_binary(Output) or is_list(Output) ->
make_text_output(State, 200, [], Output); make_text_output(State, 200, CustomHeaders, Output);
{Status, Headers, Output} {Status, Headers, Output}
when is_binary(Output) or is_list(Output) -> when is_binary(Output) or is_list(Output) ->
make_text_output(State, Status, Headers, Output); make_text_output(State, Status,
Headers ++ CustomHeaders, Output);
{Status, Reason, Headers, Output} {Status, Reason, Headers, Output}
when is_binary(Output) or is_list(Output) -> when is_binary(Output) or is_list(Output) ->
make_text_output(State, Status, Reason, Headers, Output); make_text_output(State, Status, Reason,
Headers ++ CustomHeaders, Output);
_ -> _ ->
none none
end, end,
@ -497,7 +510,7 @@ process_request(#state{request_method = Method,
end. end.
make_bad_request(State) -> make_bad_request(State) ->
make_xhtml_output(State, 400, [], make_xhtml_output(State, 400, State#state.custom_headers,
ejabberd_web:make_xhtml([#xmlel{name = <<"h1">>, ejabberd_web:make_xhtml([#xmlel{name = <<"h1">>,
attrs = [], attrs = [],
children = children =
@ -720,6 +733,11 @@ rest_dir(0, Path, <<H, T/binary>>) ->
rest_dir(0, <<H, Path/binary>>, T); rest_dir(0, <<H, Path/binary>>, T);
rest_dir(N, Path, <<_H, T/binary>>) -> rest_dir(N, Path, T). rest_dir(N, Path, <<_H, T/binary>>) -> rest_dir(N, Path, T).
expand_custom_headers(Headers) ->
lists:map(fun({K, V}) ->
{K, jlib:expand_keyword(<<"@VERSION@">>, V, ?VERSION)}
end, Headers).
%% hex_to_integer %% hex_to_integer
hex_to_integer(Hex) -> hex_to_integer(Hex) ->

View File

@ -37,7 +37,7 @@
-export([tolower/1, term_to_base64/1, base64_to_term/1, -export([tolower/1, term_to_base64/1, base64_to_term/1,
decode_base64/1, encode_base64/1, ip_to_list/1, decode_base64/1, encode_base64/1, ip_to_list/1,
hex_to_bin/1, hex_to_base64/1, hex_to_bin/1, hex_to_base64/1, expand_keyword/3,
atom_to_binary/1, binary_to_atom/1, tuple_to_binary/1, atom_to_binary/1, binary_to_atom/1, tuple_to_binary/1,
l2i/1, i2l/1, i2l/2, expr_to_term/1, term_to_expr/1]). l2i/1, i2l/1, i2l/2, expr_to_term/1, term_to_expr/1]).
@ -934,6 +934,12 @@ hex_to_bin([H1, H2 | T], Acc) ->
hex_to_base64(Hex) -> encode_base64(hex_to_bin(Hex)). hex_to_base64(Hex) -> encode_base64(hex_to_bin(Hex)).
-spec expand_keyword(binary(), binary(), binary()) -> binary().
expand_keyword(Keyword, Input, Replacement) ->
Parts = binary:split(Input, Keyword, [global]),
str:join(Parts, Replacement).
binary_to_atom(Bin) -> binary_to_atom(Bin) ->
erlang:binary_to_atom(Bin, utf8). erlang:binary_to_atom(Bin, utf8).

View File

@ -500,16 +500,14 @@ get_proc_name(ServerHost, ModuleName) ->
-spec expand_home(binary()) -> binary(). -spec expand_home(binary()) -> binary().
expand_home(Subject) -> expand_home(Input) ->
{ok, [[Home]]} = init:get_argument(home), {ok, [[Home]]} = init:get_argument(home),
Parts = binary:split(Subject, <<"@HOME@">>, [global]), jlib:expand_keyword(<<"@HOME@">>, Input, Home).
str:join(Parts, list_to_binary(Home)).
-spec expand_host(binary(), binary()) -> binary(). -spec expand_host(binary(), binary()) -> binary().
expand_host(Subject, Host) -> expand_host(Input, Host) ->
Parts = binary:split(Subject, <<"@HOST@">>, [global]), jlib:expand_keyword(<<"@HOST@">>, Input, Host).
str:join(Parts, Host).
%%-------------------------------------------------------------------- %%--------------------------------------------------------------------
%% Internal functions. %% Internal functions.
@ -835,7 +833,6 @@ http_response(Host, Code, ExtraHeaders) ->
-> {pos_integer(), [{binary(), binary()}], binary()}. -> {pos_integer(), [{binary(), binary()}], binary()}.
http_response(Host, Code, ExtraHeaders, Body) -> http_response(Host, Code, ExtraHeaders, Body) ->
ServerHeader = {<<"Server">>, <<"ejabberd ", (?VERSION)/binary>>},
CustomHeaders = CustomHeaders =
gen_mod:get_module_opt(Host, ?MODULE, custom_headers, gen_mod:get_module_opt(Host, ?MODULE, custom_headers,
fun(Headers) -> fun(Headers) ->
@ -847,10 +844,9 @@ http_response(Host, Code, ExtraHeaders, Body) ->
[]), []),
Headers = case proplists:is_defined(<<"Content-Type">>, ExtraHeaders) of Headers = case proplists:is_defined(<<"Content-Type">>, ExtraHeaders) of
true -> true ->
[ServerHeader | ExtraHeaders]; ExtraHeaders;
false -> false ->
[ServerHeader, {<<"Content-Type">>, <<"text/plain">>} | [{<<"Content-Type">>, <<"text/plain">>} | ExtraHeaders]
ExtraHeaders]
end ++ CustomHeaders, end ++ CustomHeaders,
{Code, Headers, Body}. {Code, Headers, Body}.