From 8946383a20bab472dcc89a88b56c2352e0f640a6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Daniel=20Jos=C3=A9=20Domingos=20Serrano?= Date: Sun, 28 Jun 2020 13:39:12 +0100 Subject: [PATCH] Add support for custom DISTRIBUTION metric Example of usage follows: iex> MyApp.Statix.distribution("rendering", 12, []) :ok From the Datadog documentation: > Unlike the HISTOGRAM metric type, which aggregates on the Agent during a given time interval, a DISTRIBUTION metric sends all the raw data during a time interval to Datadog, and aggregations occur server-side. Because the underlying data structure represents raw, unaggregated data, distributions provide two major features: > > - Calculation of percentile aggregations > - Customization of tagging https://docs.datadoghq.com/developers/metrics/types/?tab=distribution#definition Closes #46 --- lib/statix.ex | 25 ++++++++++++++++++++++++- lib/statix/packet.ex | 3 ++- test/statix/overriding_test.exs | 9 +++++++++ test/statix/pool_test.exs | 3 ++- test/statix_test.exs | 22 ++++++++++++++++++++++ 5 files changed, 59 insertions(+), 3 deletions(-) diff --git a/lib/statix.ex b/lib/statix.ex index 03a7bc2..572a9c0 100644 --- a/lib/statix.ex +++ b/lib/statix.ex @@ -225,6 +225,24 @@ defmodule Statix do """ @callback timing(key, value :: String.Chars.t(), options) :: on_send + @doc """ + Same as `distribution(key, value, [])`. + """ + @callback distribution(key, value :: String.Chars.t()) :: on_send + + @doc """ + Writes the given `value` to the StatsD distribution identified by `key`. + + `value` is expected in milliseconds. + + ## Examples + + iex> MyApp.Statix.distribution("rendering", 12, []) + :ok + + """ + @callback distribution(key, value :: String.Chars.t(), options) :: on_send + @doc """ Writes the given `value` to the StatsD set identified by `key`. @@ -330,6 +348,10 @@ defmodule Statix do Statix.transmit(current_statix(), :timing, key, val, options) end + def distribution(key, val, options \\ []) do + Statix.transmit(current_statix(), :distribution, key, val, options) + end + def measure(key, options \\ [], fun) when is_function(fun, 0) do {elapsed, result} = :timer.tc(fun) @@ -349,7 +371,8 @@ defmodule Statix do histogram: 3, timing: 3, measure: 3, - set: 3 + set: 3, + distribution: 3 ) end end diff --git a/lib/statix/packet.ex b/lib/statix/packet.ex index fd84ccf..49a19f4 100644 --- a/lib/statix/packet.ex +++ b/lib/statix/packet.ex @@ -38,7 +38,8 @@ defmodule Statix.Packet do gauge: "g", histogram: "h", timing: "ms", - set: "s" + set: "s", + distribution: "d" } for {name, type} <- metrics do diff --git a/test/statix/overriding_test.exs b/test/statix/overriding_test.exs index 4ae9d5e..0a4e99b 100644 --- a/test/statix/overriding_test.exs +++ b/test/statix/overriding_test.exs @@ -35,6 +35,10 @@ defmodule Statix.OverridingTest do super([key, "-overridden"], value, options) end + def distribution(key, value, options) do + super([key, "-overridden"], value, options) + end + setup do connect() end @@ -64,6 +68,11 @@ defmodule Statix.OverridingTest do assert_receive {:test_server, _, "sample-overridden:3|ms|#foo"} end + test "distribution/3" do + distribution("sample", 3, tags: ["foo"]) + assert_receive {:test_server, _, "sample-overridden:3|d|#foo"} + end + test "measure/3" do measure("sample", [tags: ["foo"]], fn -> :timer.sleep(100) diff --git a/test/statix/pool_test.exs b/test/statix/pool_test.exs index b945ab3..d98295b 100644 --- a/test/statix/pool_test.exs +++ b/test/statix/pool_test.exs @@ -18,7 +18,8 @@ defmodule Statix.PoolingTest do {:histogram, [3]}, {:timing, [3]}, {:measure, [fn -> nil end]}, - {:set, [3]} + {:set, [3]}, + {:distribution, [3]} ] |> Enum.map(fn {function, arguments} -> apply(__MODULE__, function, ["sample" | arguments]) diff --git a/test/statix_test.exs b/test/statix_test.exs index 6528f1a..14761eb 100644 --- a/test/statix_test.exs +++ b/test/statix_test.exs @@ -151,6 +151,24 @@ defmodule StatixTest do refute_received _any end + test "distribution/2,3" do + __MODULE__.distribution(["sample"], 2) + assert_receive {:test_server, _, "sample:2|d"} + + distribution("sample", 2.1) + assert_receive {:test_server, _, "sample:2.1|d"} + + distribution("sample", 3, tags: ["foo:bar", "baz"]) + assert_receive {:test_server, _, "sample:3|d|#foo:bar,baz"} + + distribution("sample", 3, sample_rate: 1.0, tags: ["foo", "bar"]) + assert_receive {:test_server, _, "sample:3|d|@1.0|#foo,bar"} + + distribution("sample", 3, sample_rate: 0.0) + + refute_received _any + end + test "port closed" do close_port() @@ -173,5 +191,9 @@ defmodule StatixTest do assert capture_log(fn -> assert {:error, :port_closed} == timing("sample", 2.5) end) =~ "timing metric \"sample\" lost value 2.5 due to port closure" + + assert capture_log(fn -> + assert {:error, :port_closed} == distribution("sample", 2.5) + end) =~ "distribution metric \"sample\" lost value 2.5 due to port closure" end end