From a0e27ffe656c3cde8e9e405630a95a0e6e2934e6 Mon Sep 17 00:00:00 2001 From: Ben Langfeld Date: Wed, 13 Aug 2014 12:21:04 -0300 Subject: [PATCH] Add mod_post_log --- mod_post_log/mod_post_log.erl | 143 ++++++++++++++++++++++++++++++++++ 1 file changed, 143 insertions(+) create mode 100644 mod_post_log/mod_post_log.erl diff --git a/mod_post_log/mod_post_log.erl b/mod_post_log/mod_post_log.erl new file mode 100644 index 0000000..c28def2 --- /dev/null +++ b/mod_post_log/mod_post_log.erl @@ -0,0 +1,143 @@ +%%%---------------------------------------------------------------------- +%%% File : mod_post_log.erl +%%% Author : Tim Stewart +%%% Purpose : POST user messages to server via HTTP +%%% Created : 02 Aug 2014 by Tim Stewart +%%% +%%% Based on mod_service_log.erl +%%%---------------------------------------------------------------------- + +-module(mod_post_log). +-author('tim@stoo.org'). + +-behaviour(gen_mod). + +-export([start/2, + stop/1, + log_user_send/3, + post_result/1]). + +-include("ejabberd.hrl"). +-include("jlib.hrl"). + +start(Host, _Opts) -> + ok = case inets:start() of + {error, {already_started, inets}} -> + ok; + ok -> + ok + end, + ejabberd_hooks:add(user_send_packet, Host, + ?MODULE, log_user_send, 50), + ok. + +stop(Host) -> + ejabberd_hooks:delete(user_send_packet, Host, + ?MODULE, log_user_send, 50), + ok. + +log_user_send(From, To, Packet) -> + ok = log_packet(From, To, Packet, From#jid.lserver). + +log_packet(From, To, {xmlelement, "message", _Attrs, _Els} = Packet, Host) -> + ok = log_message(From, To, Packet, Host); + +log_packet(_From, _To, {xmlelement, _Name, _Attrs, _Els}, _Host) -> + ok. + +log_message(From, To, {xmlelement, _Name, Attrs, _Els} = Packet, Host) -> + Type = lists:keyfind("type", 1, Attrs), + log_message_filter(Type, From, To, Packet, Host). + +log_message_filter({"type", Type}, From, To, Packet, Host) + when Type =:= "chat"; + Type =:= "groupchat" -> + log_chat(From, To, Packet, Host); +log_message_filter(_Other, _From, _To, _Packet, _Host) -> + ok. + +log_chat(From, To, {xmlelement, _Name, _Attrs, Els} = Packet, Host) -> + case get_body(Els) of + no_body -> + ok; + {ok, _Body} -> + log_chat_with_body(From, To, Packet, Host) + end. + +log_chat_with_body(_From, _To, Packet, _Host) -> + post_xml(xml:element_to_binary(Packet)). + +post_xml(Xml) -> + Ts = to_iso_8601_date(os:timestamp()), + + Body = Xml, + + Url = get_opt(url), + TsHeader = get_opt(ts_header, "X-Message-Timestamp"), + Headers = [ {TsHeader, Ts} | get_opt(headers, []) ], + io:format("Headers: ~p\n", [Headers]), + ContentType = get_opt(content_type, "text/xml"), + HttpOptions = get_opt(http_options, []), + ReqOptions = get_opt(req_options, []), + + {ok, _ReqId} = httpc:request(post, + {Url, Headers, ContentType, Body}, + HttpOptions, + [ {sync, false}, + {receiver, {?MODULE, post_result, []}} + | ReqOptions ]), + ok. + +post_result({_ReqId, {error, Reason}}) -> + report_error(Reason); +post_result({_ReqId, Result}) -> + {StatusLine, Headers, Body} = Result, + {_HttpVersion, StatusCode, ReasonPhrase} = StatusLine, + if StatusCode < 200; + StatusCode > 299 -> + ok = report_error([ {status_code, StatusCode}, + {reason_phrase, ReasonPhrase}, + {headers, Headers}, + {body, Body} ]), + ok; + true -> + ok + end. + +get_body(Els) -> + XmlElements = [ El || El <- Els, element(1, El) =:= xmlelement ], + case lists:keyfind("body", 2, XmlElements) of + false -> + no_body; + {xmlelement, "body", _, InnerEls} -> + case lists:keyfind(xmlcdata, 1, InnerEls) of + false -> + no_body; + {xmlcdata, Body} -> + {ok, Body} + end + end. + +get_opt(Opt) -> + get_opt(Opt, undefined). + +get_opt(Opt, Default) -> + gen_mod:get_module_opt(global, ?MODULE, Opt, Default). + +report_error(ReportArgs) -> + ok = error_logger:error_report([ mod_post_log_cannot_post | ReportArgs ]). + +%% Erlang now()-style timestamps are in UTC by definition, and we are +%% assuming ISO 8601 dates should be printed in UTC as well, so no +%% conversion necessary +%% +%% Example: +%% {1385,388790,334905} +%% -becomes- +%% 2013-11-25 14:13:10.334905Z +-spec to_iso_8601_date(erlang:timestamp()) -> string(). +to_iso_8601_date(Timestamp) when is_tuple(Timestamp) -> + {{Y, Mo, D}, {H, M, S}} = calendar:now_to_universal_time(Timestamp), + {_, _, US} = Timestamp, + lists:flatten(io_lib:format("~4.10.0B-~2.10.0B-~2.10.0B ~2.10.0B:~2.10.0B:~2.10.0B.~6.10.0BZ", + [Y, Mo, D, H, M, S, US])).