diff --git a/.github/workflows/hex.yaml b/.github/workflows/hex.yaml index cf646ef..dd5885e 100644 --- a/.github/workflows/hex.yaml +++ b/.github/workflows/hex.yaml @@ -7,9 +7,9 @@ on: jobs: publish: - runs-on: ubuntu-20.04 + runs-on: ubuntu-22.04 container: - image: erlang:23.2.5.0-alpine + image: erlang:26-alpine steps: - name: Prepare run: | @@ -17,7 +17,7 @@ jobs: apk --no-cache upgrade apk --no-cache add gcc git libc-dev libc-utils libgcc linux-headers make bash \ musl-dev musl-utils ncurses-dev pcre2 pkgconf scanelf wget zlib - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - name: work around for permission issue run: | git config --global --add safe.directory /__w/prometheus_diameter_collector/prometheus_diameter_collector @@ -25,5 +25,5 @@ jobs: env: HEX_API_KEY: ${{ secrets.HEX_API_KEY }} run: | - rebar3 edoc + rebar3 ex_doc rebar3 hex publish -r hexpm --yes diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index 1c5645a..159c1bb 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -11,10 +11,10 @@ on: jobs: test: name: CI - runs-on: ubuntu-20.04 + runs-on: ubuntu-22.04 strategy: matrix: - otp: [22, 23, 24, 25] + otp: [22, 23, 24, 25, 26] container: image: erlang:${{ matrix.otp }}-alpine steps: @@ -22,9 +22,9 @@ jobs: run: | apk update apk --no-cache upgrade - apk --no-cache add gcc git libc-dev libc-utils libgcc linux-headers make bash \ + apk --no-cache add gcc git libc-dev libc-utils libgcc linux-headers make bash xz \ musl-dev musl-utils ncurses-dev pcre2 pkgconf scanelf wget zlib - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 - name: Build run: rebar3 compile - name: Run tests @@ -36,18 +36,19 @@ jobs: GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} COVERALLS_FLAG_NAME: ${{ matrix.otp }} run: DEBUG=1 rebar3 as test coveralls send || /bin/true + - name: Tar Test Output + if: ${{ always() }} + run: tar -cJf ct-logs-${{ matrix.otp }}.tar.xz _build/test/logs/ - name: Archive Test Output if: ${{ always() }} - uses: actions/upload-artifact@v3 + uses: actions/upload-artifact@v4 with: name: test-output-${{ matrix.otp }} - path: | - _build/test/logs/ - !_build/test/logs/last + path: ct-logs-${{ matrix.otp }}.tar.xz slack: needs: test - runs-on: ubuntu-20.04 + runs-on: ubuntu-22.04 if: always() steps: - name: Slack notification @@ -65,7 +66,7 @@ jobs: finish: needs: test - runs-on: ubuntu-20.04 + runs-on: ubuntu-22.04 if: always() steps: - name: Coveralls Finished diff --git a/.github/workflows/review.yaml b/.github/workflows/review.yaml new file mode 100644 index 0000000..8f97c61 --- /dev/null +++ b/.github/workflows/review.yaml @@ -0,0 +1,44 @@ +name: Review + +on: + pull_request_target: + types: + - opened + - synchronize + branches: + - master + +jobs: + code-style-review: + runs-on: ubuntu-22.04 + env: + ACCESS_TOKEN: ${{ secrets.GITHUB_TOKEN }} + steps: + - + name: work around permission issue + run: | + git config --global --add safe.directory /__w/prometheus_diameter_collector/prometheus_diameter_collector + - + name: Check out repository + uses: actions/checkout@v4 + with: + ref: ${{github.event.pull_request.head.ref}} + repository: ${{github.event.pull_request.head.repo.full_name}} + - + name: install dependencies + run: | + sudo apt update + sudo apt install rebar3 emacs erlang-mode + - + name: format + run: rebar3 fmt + - + name: automated review + uses: googleapis/code-suggester@v4 + with: + command: review + pull_number: ${{ github.event.pull_request.number }} + git_dir: '.' + - + name: check + run: git diff --quiet --exit-code diff --git a/README.md b/README.md index 036f8bd..bdb0cf0 100644 --- a/README.md +++ b/README.md @@ -7,7 +7,7 @@ DIAMETER Prometheus.io collector [![Erlang Versions][erlang version badge]][gh] [Erlang Prometheus.io exporter](https://github.com/deadtrickster/prometheus.erl) for the -Erlang/OTP DIAMTER Application +Erlang/OTP DIAMETER Application Build ----- @@ -48,5 +48,5 @@ Number of requests. [coveralls]: https://coveralls.io/github/travelping/prometheus_diameter_collector [coveralls badge]: https://img.shields.io/coveralls/travelping/prometheus_diameter_collector/master.svg?style=flat-square [gh]: https://github.com/travelping/prometheus_diameter_collector/actions/workflows/main.yml -[gh badge]: https://img.shields.io/github/workflow/status/travelping/prometheus_diameter_collector/CI?style=flat-square -[erlang version badge]: https://img.shields.io/badge/erlang-22.0%20to%2025.1-blue.svg?style=flat-square +[gh badge]: https://img.shields.io/github/actions/workflow/status/travelping/prometheus_diameter_collector/main.yml?branch=master&style=flat-square +[erlang version badge]: https://img.shields.io/badge/erlang-22.0%20to%2026.2-blue.svg?style=flat-square diff --git a/rebar.config b/rebar.config index abeb159..0c2ca36 100644 --- a/rebar.config +++ b/rebar.config @@ -1,24 +1,42 @@ +%%-*-Erlang-*- + %% == Erlang Compiler == {erl_opts, [debug_info]}. %% == Dependencies == -{deps, [{prometheus, "4.9.1"}]}. +{deps, [{prometheus, "4.11.0"}]}. %% == Xref == {xref_checks, [undefined_function_calls, undefined_functions, - locals_not_used, deprecated_function_calls, - deprecated_funcqtions]}. + locals_not_used, deprecated_function_calls, + deprecated_funcqtions]}. %% == Plugins == -{plugins, [ - % @TODO: Folow https://github.com/markusn/coveralls-erl/pull/36 and use `coveralls` after release - {coveralls, {git, "https://github.com/RoadRunnr/coveralls-erl.git", {branch, "feature/git-info"}}}, - rebar3_hex] -}. +{plugins, + [ + %% @TODO: Folow https://github.com/markusn/coveralls-erl/pull/36 and use `coveralls` after release + {coveralls, {git, "https://github.com/RoadRunnr/coveralls-erl.git", {branch, "feature/git-info"}}}, + rebar3_hex, + rebar3_ex_doc, + rebar3_fmt + ]}. + +%% == ExDoc == + +{ex_doc, [ + {extras, ["README.md", "LICENSE"]}, + {main, "README.md"}, + {source_url, "https://github.com/travelping/prometheus_diameter_collector"} +]}. + +%% == Hex == +{hex, [ + {doc, #{provider => ex_doc}} +]}. %% == Cover == {cover_enabled, true}. diff --git a/rebar.config.script b/rebar.config.script index ababb5d..6cd04b4 100644 --- a/rebar.config.script +++ b/rebar.config.script @@ -2,9 +2,9 @@ case {os:getenv("GITHUB_ACTIONS"), os:getenv("GITHUB_TOKEN")} of {"true", Token} when is_list(Token) -> CONFIG1 = [{coveralls_repo_token, Token}, - {coveralls_service_job_id, os:getenv("GITHUB_RUN_ID")}, - {coveralls_commit_sha, os:getenv("GITHUB_SHA")}, - {coveralls_flag_name, os:getenv("COVERALLS_FLAG_NAME")} | CONFIG], + {coveralls_service_job_id, os:getenv("GITHUB_RUN_ID")}, + {coveralls_commit_sha, os:getenv("GITHUB_SHA")}, + {coveralls_flag_name, os:getenv("COVERALLS_FLAG_NAME")} | CONFIG], case os:getenv("GITHUB_EVENT_NAME") =:= "pull_request" andalso string:tokens(os:getenv("GITHUB_REF"), "/") of [_, "pull", PRNO, _] -> diff --git a/src/prometheus_diameter_collector.erl b/src/prometheus_diameter_collector.erl index d652ef7..ad8f4b4 100644 --- a/src/prometheus_diameter_collector.erl +++ b/src/prometheus_diameter_collector.erl @@ -18,21 +18,21 @@ -include_lib("prometheus/include/prometheus.hrl"). -export([ - deregister_cleanup/1, - collect_mf/2, collect_metrics/2, - gather/0, - gather_statistics/5 - ]). + deregister_cleanup/1, + collect_mf/2, collect_metrics/2, + gather/0, + gather_statistics/5 + ]). -import(prometheus_model_helpers, [create_mf/5, - gauge_metric/2]). + gauge_metric/2]). -define(METRIC_NAME_PREFIX, "diameter_"). -define(METRICS, [{applications, gauge, "Number of installed DIAMETER applications."}, - {connections, gauge, "Number of connections to peers."}, - {messages, gauge, "Number of requests."}, - {errors, gauge, "Number of errors."}]). + {connections, gauge, "Number of connections to peers."}, + {messages, gauge, "Number of requests."}, + {errors, gauge, "Number of errors."}]). %%==================================================================== %% Collector API @@ -47,15 +47,15 @@ collect_mf(_Registry, Callback) -> mf(Callback, {Name, Type, Help}, Stats) -> Callback(create_mf(?METRIC_NAME(Name), Help, Type, ?MODULE, - {Type, fun(S) -> maps:get(Name, S, undefined) end, Stats})), + {Type, fun(S) -> maps:get(Name, S, undefined) end, Stats})), ok. collect_metrics(_, {Type, Fun, Stats}) -> case Fun(Stats) of - M when is_map(M) -> - [metric(Type, Labels, Value) || {Labels, Value} <- maps:to_list(M)]; - _ -> - undefined + M when is_map(M) -> + [metric(Type, Labels, Value) || {Labels, Value} <- maps:to_list(M)]; + _ -> + undefined end. metric(gauge, Labels, Value) -> @@ -76,76 +76,76 @@ gather() -> Services = diameter:services(), lists:foldl( fun(SvcName, Stats) -> - Info = diameter:service_info(SvcName, [applications, peers]), - gather_service(SvcName, Info, Stats) + Info = diameter:service_info(SvcName, [applications, peers]), + gather_service(SvcName, Info, Stats) end, #{}, Services). gather_service(SvcName, Info, Stats) -> lists:foldl( fun({applications, Apps}, S0) -> - add([applications, [{svc, SvcName}]], length(Apps), S0); - ({peers, Peers}, S0) -> - Apps = proplists:get_value(applications, Info, []), - lists:foldl( - fun({PeerName, Peer}, S1) -> - gather_peer(SvcName, PeerName, Peer, Apps, S1) - end, S0, Peers) + add([applications, [{svc, SvcName}]], length(Apps), S0); + ({peers, Peers}, S0) -> + Apps = proplists:get_value(applications, Info, []), + lists:foldl( + fun({PeerName, Peer}, S1) -> + gather_peer(SvcName, PeerName, Peer, Apps, S1) + end, S0, Peers) end, Stats, Info). gather_peer(SvcName, Peer, Info, Apps, Stats) -> lists:foldl( fun({connections, C}, S0) -> - lists:foldl( - fun(X, S1) -> - gather_connection(SvcName, Peer, X, S1) - end, S0, C); - ({statistics, S}, S0) -> - gather_statistics(SvcName, Peer, S, Apps, S0) + lists:foldl( + fun(X, S1) -> + gather_connection(SvcName, Peer, X, S1) + end, S0, C); + ({statistics, S}, S0) -> + gather_statistics(SvcName, Peer, S, Apps, S0) end, Stats, Info). gather_connection(SvcName, Peer, Values, Stats) -> {_, _, State} = proplists:get_value(watchdog, Values, {undefine, undefined, unknown}), Type = case proplists:get_value(type, Values, unknown) of - accept -> responder; - connect -> initiator; - Other -> Other - end, + accept -> responder; + connect -> initiator; + Other -> Other + end, Port = proplists:get_value(port, Values, []), Connection = case proplists:get_value(module, Port, unknown) of - diameter_tcp -> tcp; - diameter_sctp -> sctp; - OtherTP -> OtherTP - end, + diameter_tcp -> tcp; + diameter_sctp -> sctp; + OtherTP -> OtherTP + end, add([connections, [{svc, SvcName}, {peer, Peer}, {type, Type}, - {state, State}, {protocol, Connection}]], 1, Stats). + {state, State}, {protocol, Connection}]], 1, Stats). gather_statistics(SvcName, Peer, S, Apps, Stats) -> lists:foldl( fun({{{_, _, 1} = Msg, Direction}, Cnt}, S1) -> - add([messages, [{svc, SvcName}, {peer, Peer}, - {direction, Direction}, - {type, msg_type(Msg)}, - {msg, msg_name(Msg, Apps)}]], Cnt, S1); - ({{Msg, Direction, {'Result-Code', RC}}, Cnt}, S1) -> - add([messages, [{svc, SvcName}, {peer, Peer}, - {direction, Direction}, - {type, msg_type(Msg)}, - {msg, msg_name(Msg, Apps)}, - {rc, RC}]], Cnt, S1); - ({{_, _, error}, Cnt}, S1) -> + add([messages, [{svc, SvcName}, {peer, Peer}, + {direction, Direction}, + {type, msg_type(Msg)}, + {msg, msg_name(Msg, Apps)}]], Cnt, S1); + ({{Msg, Direction, {'Result-Code', RC}}, Cnt}, S1) -> + add([messages, [{svc, SvcName}, {peer, Peer}, + {direction, Direction}, + {type, msg_type(Msg)}, + {msg, msg_name(Msg, Apps)}, + {rc, RC}]], Cnt, S1); + ({{_, _, error}, Cnt}, S1) -> add([errors,[{svc, SvcName}, {peer, Peer}, {error, unknown}]], Cnt, S1); - ({{Msg, Direction, Result}, Cnt}, S1) when is_atom(Result) -> - add([messages, [{svc, SvcName}, {peer, Peer}, - {direction, Direction}, - {type, msg_type(Msg)}, - {msg, msg_name(Msg, Apps)}, - {rc, Result}]], Cnt, S1); - ({Error, Cnt}, S1) when is_atom(Error) -> - add([errors,[{svc, SvcName}, {peer, Peer}, - {error, Error}]], Cnt, S1); - (_, S1) -> - S1 + ({{Msg, Direction, Result}, Cnt}, S1) when is_atom(Result) -> + add([messages, [{svc, SvcName}, {peer, Peer}, + {direction, Direction}, + {type, msg_type(Msg)}, + {msg, msg_name(Msg, Apps)}, + {rc, Result}]], Cnt, S1); + ({Error, Cnt}, S1) when is_atom(Error) -> + add([errors,[{svc, SvcName}, {peer, Peer}, + {error, Error}]], Cnt, S1); + (_, S1) -> + S1 end, Stats, S). msg_type({_, 0}) -> answer; @@ -155,33 +155,33 @@ msg_type({_, _, 1}) -> request. try_dict(Dict, {_, CmdCode, Rbit} = Cmd) -> case code:is_loaded(Dict) of - {file, _} -> - try Dict:msg_name(CmdCode, Rbit =:= 1) of - '' -> Cmd; - Name when is_atom(Name) -> Name; - _ -> Cmd - catch - _:_ -> - Cmd - end; - _ -> - Cmd + {file, _} -> + try Dict:msg_name(CmdCode, Rbit =:= 1) of + '' -> Cmd; + Name when is_atom(Name) -> Name; + _ -> Cmd + catch + _:_ -> + Cmd + end; + _ -> + Cmd end. msg_name({0, _, _} = Cmd, _Apps) -> case try_dict(diameter_gen_base_rfc6733, Cmd) of - Name when is_atom(Name) -> - Name; - _ -> - try_dict(diameter_gen_base_rfc3588, Cmd) + Name when is_atom(Name) -> + Name; + _ -> + try_dict(diameter_gen_base_rfc3588, Cmd) end; msg_name({ApplId, _, _} = Cmd, Apps) -> case lists:filter( - fun(E) -> ApplId =:= proplists:get_value(id, E, -1) end, Apps) of - [App|_] -> - try_dict(proplists:get_value(dictionary, App, undefined), Cmd); - _ -> - Cmd + fun(E) -> ApplId =:= proplists:get_value(id, E, -1) end, Apps) of + [App|_] -> + try_dict(proplists:get_value(dictionary, App, undefined), Cmd); + _ -> + Cmd end; msg_name(_, _Apps) -> unknown. diff --git a/test/prometheus_diameter_collector_SUITE.erl b/test/prometheus_diameter_collector_SUITE.erl index 13c2ad4..ddd9a2a 100644 --- a/test/prometheus_diameter_collector_SUITE.erl +++ b/test/prometheus_diameter_collector_SUITE.erl @@ -19,33 +19,33 @@ %%% ================================================================== -export([ - all/0, - groups/0, - init_per_suite/1, - end_per_suite/1 -]). + all/0, + groups/0, + init_per_suite/1, + end_per_suite/1 + ]). %%% ================================================================== %%% Common Tests Info Callbacks Exports %%% ================================================================== -export([ - diameter_applications/0, - diameter_connections/0, - diameter_messages/0, - gather_statistics/0 -]). + diameter_applications/0, + diameter_connections/0, + diameter_messages/0, + gather_statistics/0 + ]). %%% ================================================================== %%% Common Tests Exports %%% ================================================================== -export([ - diameter_applications/1, - diameter_connections/1, - diameter_messages/1, - gather_statistics/1 -]). + diameter_applications/1, + diameter_connections/1, + diameter_messages/1, + gather_statistics/1 + ]). %%% ================================================================== %%% Includes @@ -58,34 +58,34 @@ %%% ================================================================== -define(match(Guard, Expr), - ((fun () -> - case (Expr) of - Guard -> - ok; - V -> - ct:pal("MISMATCH(~s:~b, ~s)~nExpected: ~p~nActual: ~p~n", - [?FILE, ?LINE, ??Expr, ??Guard, V]), - error(badmatch) - end - end)())). + ((fun () -> + case (Expr) of + Guard -> + ok; + V -> + ct:pal("MISMATCH(~s:~b, ~s)~nExpected: ~p~nActual: ~p~n", + [?FILE, ?LINE, ??Expr, ??Guard, V]), + error(badmatch) + end + end)())). %%% ================================================================== %%% Common Tests Callbacks %%% ================================================================== all() -> - [ - {group, collector} - ]. + [ + {group, collector} + ]. groups() -> [ - {collector, [sequence], [ - diameter_applications, - diameter_connections, - diameter_messages, - gather_statistics - ]} + {collector, [sequence], [ + diameter_applications, + diameter_connections, + diameter_messages, + gather_statistics + ]} ]. init_per_suite(Config) -> @@ -132,7 +132,7 @@ gather_statistics(_Config) -> {{{unknown,0},recv,discarded},2618}, {{{0,257,0},recv,{'Result-Code',2001}},1}], R = #{messages => #{ - [{svc,testsvc}, {peer,<<"testpeer">>}, {direction,recv}, {type,answer}, {msg,'CEA'}, {rc,2001}] => 1, - [{svc,testsvc}, {peer,<<"testpeer">>}, {direction,recv}, {type,answer}, {msg,unknown}, {rc,discarded}] => 2618, - [{svc,testsvc}, {peer,<<"testpeer">>}, {direction,send}, {type,request}, {msg,'CER'}] => 1}}, + [{svc,testsvc}, {peer,<<"testpeer">>}, {direction,recv}, {type,answer}, {msg,'CEA'}, {rc,2001}] => 1, + [{svc,testsvc}, {peer,<<"testpeer">>}, {direction,recv}, {type,answer}, {msg,unknown}, {rc,discarded}] => 2618, + [{svc,testsvc}, {peer,<<"testpeer">>}, {direction,send}, {type,request}, {msg,'CER'}] => 1}}, ?match(R, prometheus_diameter_collector:gather_statistics(testsvc, <<"testpeer">>, S, [], #{})).