Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

small updates to Telemetry #833

Open
wants to merge 6 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 5 additions & 4 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ jobs:
STRIPE_MOCK_VERSION: 0.144.0
STRIPE_SECRET_KEY: non_empty_string
SKIP_STRIPE_MOCK_RUN: true
SHELL: bash
runs-on: ubuntu-20.04
name: Test (OTP ${{matrix.otp}} / Elixir ${{matrix.elixir}})
strategy:
Expand All @@ -38,10 +39,9 @@ jobs:
with:
otp-version: ${{ matrix.otp }}
elixir-version: ${{ matrix.elixir }}
- name: Mix Dependencies
run: mix deps.get
- name: Test
run: mix test
- run: mix deps.get
- run: mix compile
- run: mix test --trace

lint:
runs-on: ubuntu-20.04
Expand Down Expand Up @@ -76,3 +76,4 @@ jobs:
- run: mix compile --warnings-as-errors
- run: mix dialyzer --halt-exit-status
- run: mix deps.unlock --check-unused
- run: mix format --check-formatted
14 changes: 11 additions & 3 deletions lib/stripe/api.ex
Original file line number Diff line number Diff line change
Expand Up @@ -376,14 +376,22 @@ defmodule Stripe.API do
end

defp do_perform_request_and_retry(method, url, headers, body, opts, {:attempts, attempts}) do
telemetry_meta = %{
endpoint: URI.parse(url).path,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
endpoint: URI.parse(url).path,
http_url: URI.parse(url),

Pass the entire URL, let processors to filter the info, HTTP OTEL packages would do something around the line of https://github.com/open-telemetry/opentelemetry-erlang-contrib/blob/3f13bc8b2042071b723d0ea8c7b118fd10f42d26/instrumentation/opentelemetry_req/lib/opentelemetry_req.ex#L143-L148

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

okay added and I renamed original field to stripe_api_endpoint. I can remove it because you're right that it can be extracted from http_url via processor, I kept it for better out-ot-the-box experience.

method: method,
robuye marked this conversation as resolved.
Show resolved Hide resolved
attempt: attempts,
robuye marked this conversation as resolved.
Show resolved Hide resolved
stripe_api_version: headers["Stripe-Version"],
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
stripe_api_version: headers["Stripe-Version"],
http_headers: headers,

status: nil
robuye marked this conversation as resolved.
Show resolved Hide resolved
}

response =
:telemetry.span(~w[stripe request]a, %{url: url, method: method}, fn ->
:telemetry.span(~w[stripe request]a, telemetry_meta, fn ->
case http_module().request(method, url, Map.to_list(headers), body, opts) do
yordis marked this conversation as resolved.
Show resolved Hide resolved
{:ok, status, _, _} = resp ->
{resp, %{status: status}}
{resp, %{telemetry_meta | status: status}}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
{resp, %{telemetry_meta | status: status}}
{resp, %{telemetry_meta | http_status_code: status, http_response_content_length: http_response_content_length }}

If content-length is present add http_response_content_length


error ->
{error, %{}}
{error, telemetry_meta}
end
end)

Expand Down
56 changes: 32 additions & 24 deletions lib/stripe/webhook_plug.ex
Original file line number Diff line number Diff line change
Expand Up @@ -159,30 +159,38 @@ if Code.ensure_loaded?(Plug) do
end

defp handle_event!(handler, %Stripe.Event{} = event) do
case handler.handle_event(event) do
{:ok, _} ->
:ok

:ok ->
:ok

{:error, reason} when is_binary(reason) ->
{:handle_error, reason}

{:error, reason} when is_atom(reason) ->
{:handle_error, Atom.to_string(reason)}

:error ->
{:handle_error, ""}

resp ->
raise """
#{inspect(handler)}.handle_event/1 returned an invalid response. Expected {:ok, term}, :ok, {:error, reason} or :error
Got: #{inspect(resp)}

Event data: #{inspect(event)}
"""
end
telemetry_meta = %{event: event.type, handler_status: nil}

:telemetry.span(~w[stripe webhook]a, telemetry_meta, fn ->
case handler.handle_event(event) do
{:ok, _} ->
:ok

:ok ->
:ok

{:error, reason} when is_binary(reason) ->
{:handle_error, reason}

{:error, reason} when is_atom(reason) ->
{:handle_error, Atom.to_string(reason)}

:error ->
{:handle_error, ""}

resp ->
raise """
#{inspect(handler)}.handle_event/1 returned an invalid response. Expected {:ok, term}, :ok, {:error, reason} or :error
Got: #{inspect(resp)}

Event data: #{inspect(event)}
"""
end
|> then(fn
:ok -> {:ok, %{telemetry_meta | handler_status: :ok}}
result -> {result, %{telemetry_meta | handler_status: :error}}
end)
end)
end

defp parse_secret!({m, f, a}), do: apply(m, f, a)
Expand Down
39 changes: 39 additions & 0 deletions test/stripe/api_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,10 @@ defmodule Stripe.APITest do
import Mox
use Stripe.StripeCase

def telemetry_handler_fn(name, measurements, metadata, _config) do
send(self(), {:telemetry_event, name, measurements, metadata})
end

test "works with non existent responses without issue" do
{:error, %Stripe.Error{extra: %{http_status: 404}}} =
Stripe.API.request(%{}, :get, "/", %{}, [])
Expand Down Expand Up @@ -87,6 +91,41 @@ defmodule Stripe.APITest do
end
end

describe "telemetry" do
test "requests emit :start, :stop telemetry events", %{test: test} do
:telemetry.attach_many(
"#{test}",
[[:stripe, :request, :start], [:stripe, :request, :stop]],
&__MODULE__.telemetry_handler_fn/4,
nil
)

%{query: ~s|email: "[email protected]"|}
|> Stripe.API.request(:get, "/v1/customers/search", %{}, [])

assert_received({
:telemetry_event,
[:stripe, :request, :start],
%{monotonic_time: _},
%{telemetry_span_context: _}
})

assert_received({
:telemetry_event,
[:stripe, :request, :stop],
%{monotonic_time: _, duration: _},
%{
telemetry_span_context: _,
endpoint: "/v1/customers/search",
attempt: 0,
method: :get,
status: 200,
stripe_api_version: _
}
})
end
end

@tag :skip
test "gets default api version" do
Stripe.API.request(%{}, :get, "products", %{}, [])
Expand Down
9 changes: 7 additions & 2 deletions test/stripe/util_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -37,8 +37,13 @@ defmodule Stripe.UtilTest do

assert object_name_to_module("billing_portal.session") == Stripe.BillingPortal.Session
assert object_name_to_module("checkout.session") == Stripe.Checkout.Session
assert object_name_to_module("identity.verification_report") == Stripe.Identity.VerificationReport
assert object_name_to_module("identity.verification_session") == Stripe.Identity.VerificationSession

assert object_name_to_module("identity.verification_report") ==
Stripe.Identity.VerificationReport

assert object_name_to_module("identity.verification_session") ==
Stripe.Identity.VerificationSession

assert object_name_to_module("issuing.authorization") == Stripe.Issuing.Authorization
assert object_name_to_module("issuing.card") == Stripe.Issuing.Card
assert object_name_to_module("issuing.cardholder") == Stripe.Issuing.Cardholder
Expand Down
97 changes: 96 additions & 1 deletion test/stripe/webhook_plug_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ defmodule Stripe.WebhookPlugTest do
use Plug.Test
alias Stripe.WebhookPlug

@valid_payload ~S({"object": "event"})
@valid_payload ~S({"object": "event", "type": "customer.updated"})
@secret "secret"

@opts WebhookPlug.init(
Expand Down Expand Up @@ -46,6 +46,10 @@ defmodule Stripe.WebhookPlugTest do

def get_value(:secret), do: @secret

def telemetry_handler_fn(name, measurements, metadata, _config) do
send(self(), {:telemetry_event, name, measurements, metadata})
end

defp generate_signature_header(payload) do
timestamp = System.system_time(:second)

Expand Down Expand Up @@ -185,5 +189,96 @@ defmodule Stripe.WebhookPlugTest do
WebhookPlug.call(conn, opts)
end
end

test "emits :start, :exception telemetry events on exception", %{conn: conn, test: test} do
:telemetry.attach_many(
"#{test}",
[[:stripe, :webhook, :start], [:stripe, :webhook, :exception]],
&__MODULE__.telemetry_handler_fn/4,
nil
)

opts =
WebhookPlug.init(
at: "/webhook/stripe",
handler: __MODULE__.BadHandler,
secret: @secret
)

assert_raise RuntimeError, fn ->
WebhookPlug.call(conn, opts)
end

assert_received({
:telemetry_event,
[:stripe, :webhook, :start],
%{monotonic_time: _},
%{telemetry_span_context: _}
})

assert_received({
:telemetry_event,
[:stripe, :webhook, :exception],
%{monotonic_time: _, duration: _},
%{kind: _, reason: _, stacktrace: _, telemetry_span_context: _, event: "customer.updated"}
})
end

test "emits :start, :stop telemetry events on soft failure", %{conn: conn, test: test} do
:telemetry.attach_many(
"#{test}",
[[:stripe, :webhook, :start], [:stripe, :webhook, :stop]],
&__MODULE__.telemetry_handler_fn/4,
nil
)

opts =
WebhookPlug.init(
at: "/webhook/stripe",
handler: __MODULE__.ErrorTupleAtomHandler,
secret: @secret
)

WebhookPlug.call(conn, opts)

assert_received({
:telemetry_event,
[:stripe, :webhook, :start],
%{monotonic_time: _},
%{telemetry_span_context: _}
})

assert_received({
:telemetry_event,
[:stripe, :webhook, :stop],
%{monotonic_time: _, duration: _},
%{handler_status: :error, telemetry_span_context: _}
})
end

test "emits :start, :stop telemetry events on success", %{conn: conn, test: test} do
:telemetry.attach_many(
"#{test}",
[[:stripe, :webhook, :start], [:stripe, :webhook, :stop]],
&__MODULE__.telemetry_handler_fn/4,
nil
)

WebhookPlug.call(conn, @opts)

assert_received({
:telemetry_event,
[:stripe, :webhook, :start],
%{monotonic_time: _},
%{telemetry_span_context: _}
})

assert_received({
:telemetry_event,
[:stripe, :webhook, :stop],
%{monotonic_time: _, duration: _},
%{handler_status: :ok, event: "customer.updated", telemetry_span_context: _}
})
end
end
end
25 changes: 15 additions & 10 deletions test/support/stripe_case.ex
Original file line number Diff line number Diff line change
Expand Up @@ -6,14 +6,27 @@ defmodule Stripe.StripeCase do
use ExUnit.CaseTemplate

def assert_stripe_requested(expected_method, path, extra \\ []) do
expected_url = build_url(path, Keyword.get(extra, :query))
expected_params = Keyword.get(extra, :query, %{})
expected_path = URI.parse(path).path
expected_body = Keyword.get(extra, :body)
expected_headers = Keyword.get(extra, :headers)

assert_received({method, url, headers, body, _})

actual_uri = URI.parse(url)

actual_query_params =
to_string(actual_uri.query)
|> URI.query_decoder()
|> Enum.into(%{})

Enum.each(expected_params, fn {key, value} ->
actual_val = Map.get(actual_query_params, to_string(key))
assert actual_val == to_string(value)
end)

assert expected_method == method
assert expected_url == url
assert expected_path == actual_uri.path

assert_stripe_request_body(expected_body, body)
assert_stripe_request_headers(expected_headers, headers)
Expand Down Expand Up @@ -51,14 +64,6 @@ defmodule Stripe.StripeCase do
assert body == Stripe.URI.encode_query(expected_body)
end

defp build_url(path, nil) do
stripe_base_url() <> path
end

defp build_url(path, query_params) do
stripe_base_url() <> path <> "?" <> URI.encode_query(query_params)
end

defmodule HackneyMock do
@doc """
Send message to the owning process for each request so we can assert that
Expand Down
2 changes: 1 addition & 1 deletion test/support/stripe_mock_test.exs
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ defmodule Stripe.StripeMockTest do

defp assert_port_open(port) do
delay()
assert {:ok, socket} = :gen_tcp.connect('localhost', port, [])
assert {:ok, socket} = :gen_tcp.connect(~c"localhost", port, [])
:gen_tcp.close(socket)
end

Expand Down
Loading