-
-
Notifications
You must be signed in to change notification settings - Fork 198
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
Add SpanProcessor for OpenTelemetry #875
Open
solnic
wants to merge
25
commits into
master
Choose a base branch
from
874-add-spanprocessor-for-otel
base: master
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Open
Changes from all commits
Commits
Show all changes
25 commits
Select commit
Hold shift + click to select a range
0710007
Add SpanProcessor for OpenTelemetry
solnic 178825d
Move opentelemetry config to test env
solnic 5eb88c2
Add tracing config option
solnic 4c96544
Minor formatting fix
solnic 71937b0
Mark SpanRecord as @moduledoc false
solnic 894a30b
Match otel span when building SpanRecord
solnic dc243aa
Clarify usage of optional integrations
solnic b219c75
Skip loading Sampler if otel_sampler is not loaded
solnic ece52cd
Specify sampler behaviour
solnic 3ccf675
Use Sentry.Case in SpanStorage tests
solnic cd999ef
No need to check if :span_storage table exists when cleaning
solnic de16794
Simplify span storage clean up in SpanProcessor tests
solnic f3ebad1
Simplify SpanStorage.start_link/1
solnic 556ee0e
Turn SpanStorage into a struct
solnic 9186bca
Move handle_info up
solnic e5f3858
Simplify how root spans are stored
solnic bbe4303
Extract SpanStorage.insert_root_span/2
solnic 31e19ec
Rename remove_span to remove_root_span
solnic 26921f6
Reuse get_root_span
solnic 2217a55
Split update_span for root and child spans
solnic c7a2309
Make SamplerTest async
solnic 8767d20
fixup! Simplify how root spans are stored
solnic 0c81602
Simplify stale span cleanup
solnic 5759f89
Refactor SpanStorage to allow custom table names
solnic 65c039c
No need to guard when creating ets table
solnic File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
if Code.ensure_loaded?(:otel_sampler) do | ||
defmodule Sentry.OpenTelemetry.Sampler do | ||
@moduledoc false | ||
|
||
@behaviour :otel_sampler | ||
|
||
def setup(config) do | ||
config | ||
end | ||
|
||
def description(_) do | ||
"SentrySampler" | ||
end | ||
|
||
def should_sample( | ||
_ctx, | ||
_trace_id, | ||
_links, | ||
span_name, | ||
_span_kind, | ||
_attributes, | ||
config | ||
) do | ||
if span_name in config[:drop] do | ||
{:drop, [], []} | ||
else | ||
{:record_and_sample, [], []} | ||
end | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,199 @@ | ||
if Code.ensure_loaded?(OpenTelemetry) do | ||
defmodule Sentry.OpenTelemetry.SpanProcessor do | ||
@moduledoc false | ||
|
||
require OpenTelemetry.SemConv.ClientAttributes, as: ClientAttributes | ||
require OpenTelemetry.SemConv.Incubating.DBAttributes, as: DBAttributes | ||
require OpenTelemetry.SemConv.Incubating.HTTPAttributes, as: HTTPAttributes | ||
require OpenTelemetry.SemConv.Incubating.URLAttributes, as: URLAttributes | ||
require OpenTelemetry.SemConv.Incubating.MessagingAttributes, as: MessagingAttributes | ||
@behaviour :otel_span_processor | ||
|
||
require Logger | ||
|
||
alias Sentry.{Transaction, OpenTelemetry.SpanStorage, OpenTelemetry.SpanRecord} | ||
alias Sentry.Interfaces.Span | ||
|
||
@impl true | ||
def on_start(_ctx, otel_span, _config) do | ||
span_record = SpanRecord.new(otel_span) | ||
|
||
SpanStorage.store_span(span_record) | ||
|
||
otel_span | ||
end | ||
|
||
@impl true | ||
def on_end(otel_span, _config) do | ||
span_record = SpanRecord.new(otel_span) | ||
|
||
SpanStorage.update_span(span_record) | ||
|
||
if span_record.parent_span_id == nil do | ||
root_span_record = SpanStorage.get_root_span(span_record.span_id) | ||
child_span_records = SpanStorage.get_child_spans(span_record.span_id) | ||
transaction = build_transaction(root_span_record, child_span_records) | ||
|
||
result = | ||
case Sentry.send_transaction(transaction) do | ||
solnic marked this conversation as resolved.
Show resolved
Hide resolved
|
||
{:ok, _id} -> | ||
true | ||
|
||
:ignored -> | ||
true | ||
|
||
{:error, error} -> | ||
Logger.error("Failed to send transaction to Sentry: #{inspect(error)}") | ||
{:error, :invalid_span} | ||
end | ||
|
||
:ok = SpanStorage.remove_root_span(span_record.span_id) | ||
|
||
result | ||
else | ||
true | ||
end | ||
end | ||
|
||
@impl true | ||
def force_flush(_config) do | ||
:ok | ||
end | ||
|
||
defp build_transaction(root_span_record, child_span_records) do | ||
root_span = build_span(root_span_record) | ||
child_spans = Enum.map(child_span_records, &build_span(&1)) | ||
|
||
Transaction.new(%{ | ||
span_id: root_span.span_id, | ||
transaction: transaction_name(root_span_record), | ||
transaction_info: %{source: :custom}, | ||
start_timestamp: root_span_record.start_time, | ||
timestamp: root_span_record.end_time, | ||
contexts: %{ | ||
trace: build_trace_context(root_span_record), | ||
otel: build_otel_context(root_span_record) | ||
}, | ||
spans: child_spans | ||
}) | ||
end | ||
|
||
defp transaction_name( | ||
%{attributes: %{unquote(to_string(MessagingAttributes.messaging_system())) => :oban}} = | ||
span_record | ||
) do | ||
span_record.attributes["oban.job.worker"] | ||
end | ||
|
||
defp transaction_name(span_record), do: span_record.name | ||
|
||
defp build_trace_context(span_record) do | ||
{op, description} = get_op_description(span_record) | ||
|
||
%{ | ||
trace_id: span_record.trace_id, | ||
span_id: span_record.span_id, | ||
parent_span_id: span_record.parent_span_id, | ||
op: op, | ||
description: description, | ||
origin: span_record.origin, | ||
data: span_record.attributes | ||
} | ||
end | ||
|
||
defp build_otel_context(span_record), do: span_record.attributes | ||
|
||
defp get_op_description( | ||
%{ | ||
attributes: %{ | ||
unquote(to_string(HTTPAttributes.http_request_method())) => http_request_method | ||
} | ||
} = span_record | ||
) do | ||
op = "http.#{span_record.kind}" | ||
|
||
client_address = | ||
Map.get(span_record.attributes, to_string(ClientAttributes.client_address())) | ||
|
||
url_path = Map.get(span_record.attributes, to_string(URLAttributes.url_path())) | ||
|
||
description = | ||
to_string(http_request_method) <> | ||
((client_address && " from #{client_address}") || "") <> | ||
((url_path && " #{url_path}") || "") | ||
|
||
{op, description} | ||
end | ||
|
||
defp get_op_description( | ||
%{attributes: %{unquote(to_string(DBAttributes.db_system())) => _db_system}} = | ||
span_record | ||
) do | ||
db_query_text = Map.get(span_record.attributes, "db.statement") | ||
|
||
{"db", db_query_text} | ||
end | ||
|
||
defp get_op_description(%{ | ||
attributes: | ||
%{unquote(to_string(MessagingAttributes.messaging_system())) => :oban} = attributes | ||
}) do | ||
{"queue.process", attributes["oban.job.worker"]} | ||
end | ||
|
||
defp get_op_description(span_record) do | ||
{span_record.name, span_record.name} | ||
end | ||
|
||
defp build_span(span_record) do | ||
{op, description} = get_op_description(span_record) | ||
|
||
%Span{ | ||
op: op, | ||
description: description, | ||
start_timestamp: span_record.start_time, | ||
timestamp: span_record.end_time, | ||
trace_id: span_record.trace_id, | ||
span_id: span_record.span_id, | ||
parent_span_id: span_record.parent_span_id, | ||
origin: span_record.origin, | ||
data: Map.put(span_record.attributes, "otel.kind", span_record.kind), | ||
status: span_status(span_record) | ||
} | ||
end | ||
|
||
defp span_status(%{ | ||
attributes: %{ | ||
unquote(to_string(HTTPAttributes.http_response_status_code())) => | ||
http_response_status_code | ||
} | ||
}) do | ||
to_status(http_response_status_code) | ||
end | ||
|
||
defp span_status(_span_record), do: nil | ||
|
||
# WebSocket upgrade spans doesn't have a HTTP status | ||
defp to_status(nil), do: nil | ||
|
||
defp to_status(status) when status in 200..299, do: "ok" | ||
|
||
for {status, string} <- %{ | ||
400 => "invalid_argument", | ||
401 => "unauthenticated", | ||
403 => "permission_denied", | ||
404 => "not_found", | ||
409 => "already_exists", | ||
429 => "resource_exhausted", | ||
499 => "cancelled", | ||
500 => "internal_error", | ||
501 => "unimplemented", | ||
503 => "unavailable", | ||
504 => "deadline_exceeded" | ||
} do | ||
defp to_status(unquote(status)), do: unquote(string) | ||
end | ||
|
||
defp to_status(_any), do: "unknown_error" | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,74 @@ | ||
if Code.ensure_loaded?(OpenTelemetry) do | ||
defmodule Sentry.OpenTelemetry.SpanRecord do | ||
solnic marked this conversation as resolved.
Show resolved
Hide resolved
|
||
@moduledoc false | ||
|
||
require Record | ||
require OpenTelemetry | ||
|
||
@fields Record.extract(:span, from_lib: "opentelemetry/include/otel_span.hrl") | ||
Record.defrecordp(:span, @fields) | ||
solnic marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
defstruct @fields ++ [:origin] | ||
|
||
def new(span() = otel_span) do | ||
otel_attrs = span(otel_span) | ||
|
||
{:attributes, _, _, _, attributes} = otel_attrs[:attributes] | ||
|
||
origin = | ||
case otel_attrs[:instrumentation_scope] do | ||
{:instrumentation_scope, origin, _version, _} -> | ||
origin | ||
|
||
_ -> | ||
:undefined | ||
end | ||
|
||
attrs = | ||
otel_attrs | ||
|> Keyword.delete(:attributes) | ||
|> Keyword.merge( | ||
trace_id: cast_trace_id(otel_attrs[:trace_id]), | ||
span_id: cast_span_id(otel_attrs[:span_id]), | ||
parent_span_id: cast_span_id(otel_attrs[:parent_span_id]), | ||
origin: origin, | ||
start_time: cast_timestamp(otel_attrs[:start_time]), | ||
end_time: cast_timestamp(otel_attrs[:end_time]), | ||
attributes: normalize_attributes(attributes) | ||
) | ||
|> Map.new() | ||
|
||
struct(__MODULE__, attrs) | ||
end | ||
|
||
defp normalize_attributes(attributes) do | ||
Enum.map(attributes, fn {key, value} -> | ||
{to_string(key), value} | ||
end) | ||
|> Map.new() | ||
end | ||
|
||
defp cast_span_id(nil), do: nil | ||
defp cast_span_id(:undefined), do: nil | ||
defp cast_span_id(span_id), do: bytes_to_hex(span_id, 16) | ||
|
||
defp cast_trace_id(trace_id), do: bytes_to_hex(trace_id, 32) | ||
|
||
defp cast_timestamp(:undefined), do: nil | ||
defp cast_timestamp(nil), do: nil | ||
|
||
defp cast_timestamp(timestamp) do | ||
nano_timestamp = OpenTelemetry.timestamp_to_nano(timestamp) | ||
{:ok, datetime} = DateTime.from_unix(div(nano_timestamp, 1_000_000), :millisecond) | ||
|
||
DateTime.to_iso8601(datetime) | ||
end | ||
|
||
defp bytes_to_hex(bytes, length) do | ||
case(:otel_utils.format_binary_string("~#{length}.16.0b", [bytes])) do | ||
{:ok, result} -> result | ||
{:error, _} -> raise "Failed to convert bytes to hex: #{inspect(bytes)}" | ||
end | ||
end | ||
end | ||
end |
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The way you've implemented it, we can do nothing in
on_start
since theon_end
logic is sufficient to build the span tree and transaction.This will also simplify your
SpanStorage.update_span
where you can remove the unnecessary deletes.(This is also how we've done it in the new python and JS span processors.)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@sl0thentr0py I'll dig into this more but when I make on_start a no-op the tests are no longer passing 🙃