This repository has been archived by the owner on Sep 27, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 96
/
Copy pathlogplex_redis_writer.erl
126 lines (115 loc) · 5.26 KB
/
logplex_redis_writer.erl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
%% Copyright (c) 2010 Jacob Vorreuter <[email protected]>
%%
%% Permission is hereby granted, free of charge, to any person
%% obtaining a copy of this software and associated documentation
%% files (the "Software"), to deal in the Software without
%% restriction, including without limitation the rights to use,
%% copy, modify, merge, publish, distribute, sublicense, and/or sell
%% copies of the Software, and to permit persons to whom the
%% Software is furnished to do so, subject to the following
%% conditions:
%%
%% The above copyright notice and this permission notice shall be
%% included in all copies or substantial portions of the Software.
%%
%% THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
%% EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
%% OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
%% NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
%% HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
%% WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
%% FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
%% OTHER DEALINGS IN THE SOFTWARE.
-module(logplex_redis_writer).
-export([start_link/2, init/3, loop/3]).
-include("logplex.hrl").
-include("logplex_logging.hrl").
%% API functions
start_link(BufferPid, RedisOpts) ->
?INFO("at=start_link redis_opts=~p~n", [RedisOpts]),
proc_lib:start_link(?MODULE, init, [self(), BufferPid, RedisOpts], 5000).
init(Parent, BufferPid, RedisOpts) ->
?INFO("at=init buffer_pid=~p parent=~p~n", [BufferPid, Parent]),
logplex_queue:register(BufferPid, self()),
proc_lib:init_ack(Parent, {ok, self()}),
random:seed(os:timestamp()),
connect_with_backoff(BufferPid, RedisOpts, 0).
loop(BufferPid, Socket, RedisOpts) when is_port(Socket) ->
verify_open_connection(Socket),
write_queued_logs(BufferPid, Socket),
check_for_stop_signal(),
?MODULE:loop(BufferPid, Socket, RedisOpts).
verify_open_connection(Socket) ->
case gen_tcp:recv(Socket, 0, 0) of
{ok, _} -> ok;
{error, timeout} -> ok;
{error, closed} ->
throttled_restart(closed)
end.
write_queued_logs(BufferPid, Socket) ->
case catch logplex_queue:out(BufferPid, 100) of
{'EXIT', {noproc, _}} ->
exit(normal);
timeout -> ok;
{NumBatches, [{_, _} | _] = BatchCmds} when is_integer(NumBatches) ->
Res = [ case gen_tcp:send(Socket, Cmd) of
ok ->
logplex_stats:incr(message_processed, BatchSize),
logplex_realtime:incr('message.processed', BatchSize),
logplex_stats:incr(message_batch_processed),
logplex_realtime:incr('message_batch.processed'),
1;
Err ->
?INFO("event=send_batch error=~p", [Err]),
0
end || {Cmd, BatchSize} <- BatchCmds ],
case NumBatches - lists:sum(Res) of
0 -> ok;
N ->
logplex_stats:incr(message_batch_failed, N),
logplex_realtime:incr('message_batch.failed', N),
throttled_restart(batch_processing_failed)
end;
{NumItems, Logs} when is_list(Logs), is_integer(NumItems) ->
case gen_tcp:send(Socket, Logs) of
ok ->
logplex_stats:incr(message_processed, NumItems),
logplex_realtime:incr('message.processed', NumItems);
{error, closed} ->
throttled_restart(closed);
Err ->
?INFO("event=send result=~p", [Err]),
exit({error, Err})
end;
Else ->
?WARN("event=queue_out result=~p", [Else]),
exit({error, {unexpected_result, Else}})
end.
check_for_stop_signal() ->
receive stop -> exit(normal) after 0 -> ok end.
throttled_restart(Reason) ->
throttled_restart(Reason, 5000).
throttled_restart(Reason, Delay) ->
?INFO("event=send result=~p exit_in=~pms", [Reason, Delay]),
timer:sleep(Delay),
exit({error, closed}).
connect_with_backoff(BufferPid, _RedisOpts, 10 = _MaxAttempts) ->
?WARN("at=connect_with_backoff msg=connect_attempts_exhausted buffer_pid=~p", [BufferPid]),
exit({error, connect_attempts_exhausted});
connect_with_backoff(BufferPid, RedisOpts, Attempt) ->
case open_socket(RedisOpts) of
{error, Reason} ->
?INFO("at=connect_with_backoff msg=failed_to_open_socket buffer_pid=~p attempt=~p reason=~p",
[BufferPid, Attempt, Reason]),
BaseBackoff = logplex_app:config(redis_buffer_base_backoff, timer:seconds(5)),
Jitter = case BaseBackoff of _ when BaseBackoff > 0 -> random:uniform(BaseBackoff); _ -> 0 end,
timer:sleep(trunc((Attempt * BaseBackoff) + Jitter)),
connect_with_backoff(BufferPid, RedisOpts, Attempt + 1);
{ok, Socket} when is_port(Socket) ->
loop(BufferPid, Socket, RedisOpts)
end.
open_socket(Opts) ->
Ip = proplists:get_value(ip, Opts),
Port = proplists:get_value(port, Opts),
Pass = proplists:get_value(pass, Opts),
redis:connect(Ip, Port, Pass).