blob: 027402b5152ca1f3c42810f97ec4a8caa5fc1d79 [file] [log] [blame]
% Licensed under the Apache License, Version 2.0 (the "License"); you may not
% use this file except in compliance with the License. You may obtain a copy of
% the License at
%
% http://www.apache.org/licenses/LICENSE-2.0
%
% Unless required by applicable law or agreed to in writing, software
% distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
% WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
% License for the specific language governing permissions and limitations under
% the License.
-module(couch_prometheus_e2e_tests).
-include_lib("couch/include/couch_eunit.hrl").
-define(USER, "prometheus_test_admin").
-define(PASS, "pass").
-define(AUTH, {basic_auth, {?USER, ?PASS}}).
-define(PROM_PORT, "17986").
-define(CONTENT_JSON, {"Content-Type", "application/json"}).
e2e_test_() ->
{
"With dedicated port",
{
setup,
fun() ->
setup_prometheus(true)
end,
fun(Ctx) ->
test_util:stop_couch(Ctx)
end,
{
foreach,
fun() ->
mochiweb_socket_server:get(chttpd, port)
end,
[
?TDEF_FE(t_chttpd_port),
?TDEF_FE(t_prometheus_port),
?TDEF_FE(t_metric_updated)
]
}
}
}.
reject_test_() ->
{
"Without dedicated port",
{
setup,
fun() ->
setup_prometheus(false)
end,
fun(Ctx) ->
test_util:stop_couch(Ctx)
end,
{
foreach,
fun() ->
?PROM_PORT
end,
[
?TDEF_FE(t_reject_prometheus_port)
]
}
}
}.
setup_prometheus(WithAdditionalPort) ->
Ctx = test_util:start_couch([chttpd]),
Persist = false,
Hashed = couch_passwords:hash_admin_password(?PASS),
ok = config:set("admins", ?USER, binary_to_list(Hashed), Persist),
ok = config:set_integer("stats", "interval", 1, Persist),
ok = config:set_integer("prometheus", "interval", 1, Persist),
ok = config:set_boolean(
"prometheus",
"additional_port",
WithAdditionalPort,
Persist
),
ok = config:set("prometheus", "port", ?PROM_PORT, Persist),
% It's already started by default, so restart to pick up config
ok = application:stop(couch_prometheus),
ok = application:start(couch_prometheus),
Ctx.
t_chttpd_port(Port) ->
{ok, RC, _, _} = test_request:get(node_local_url(Port), [?CONTENT_JSON, ?AUTH]),
?assertEqual(200, RC).
t_prometheus_port(_) ->
Url = node_local_url(?PROM_PORT),
{ok, RC1, _, _} = test_request:get(Url, [?CONTENT_JSON, ?AUTH]),
?assertEqual(200, RC1),
% Since this port doesn't require auth, this should work
{ok, RC2, _, _} = test_request:get(Url, [?CONTENT_JSON]),
?assertEqual(200, RC2).
t_reject_prometheus_port(Port) ->
Response = test_request:get(node_local_url(Port), [?CONTENT_JSON, ?AUTH]),
?assertEqual({error, {conn_failed, {error, econnrefused}}}, Response).
t_metric_updated(Port) ->
% The passage of time should increment this metric
Metric = "couchdb_uptime_seconds",
Url = node_local_url(Port),
% We may need to wait until the metric appears
InitialValue = test_util:wait(
fun() ->
Stats = get_stats(Url),
case metric_value(Stats, Metric) of
not_found -> wait;
Val -> Val
end
end
),
test_util:wait(
fun() ->
NewValue = metric_value(get_stats(Url), Metric),
case NewValue > InitialValue of
true -> ok;
false -> wait
end
end
).
node_local_url(Port) ->
Addr = config:get("chttpd", "bind_address", "127.0.0.1"),
lists:concat(["http://", Addr, ":", Port, "/_node/_local/_prometheus"]).
get_stats(Url) ->
{ok, _, _, Body} = test_request:get(Url, [?CONTENT_JSON, ?AUTH]),
Body.
metric_value(StatsBin, Metric) ->
% Prefix metric with newline to avoid matching lines starting with "# TYPE"
case string:find(StatsBin, "\n" ++ Metric, trailing) of
nomatch ->
not_found;
Leading ->
Trimmed = string:trim(Leading, leading),
[Line, _Rest] = string:split(Trimmed, "\n"),
[_MetricBin, Value] = string:split(Line, " "),
binary_to_integer(Value)
end.