* Use test matrix from file * Only check formatting on specific Elixir version * Use latest patch version of each Elixir/OTP release in test matrix * Test on Elixir 1.15 and OTP 26 * Run formatter on opentelemetry_httpoison * Run formatter on opentelemetry_phoenix * Run formatter on opentelemetry_tesla * Fix building opentelemetry_ecto on Elixir 1.15 Upgraded deps to fix ssl_verify_fun not compiling * Fix building opentelemetry_dataloader on Elixir 1.15 Upgraded deps to fix ssl_verify_fun and ecto_sql not compiling * Upgrade opentelemetry_finch to build on Elixir 1.15 * Upgrade opentelemetry_httpoison deps to build on 1.15 * Upgrade opentelemetry_nebulex to build on Elixir 1.15 * Upgrade opentelemetry_oban to build on Elixir 1.15 * Upgrade opentelemetry_phoenix deps to build on 1.15 * Upgrade opentelemetry_redix deps to build on 1.15 * Fix warning about <> being ambiguous * Fix assertion on attributes keys These are always atoms, not strings. * Upgrade ssl_verify_fun in opentelemetry_telemetry * Deterministically sort keys before asserting in tests * Upgrade opentelemetry_process_propogator to build on Elixir 1.15 * Run mix format on opentelemetry_process_propogator * Assert keys are atoms, not strings * Use matrix.os to define runs-on parameter * Pin test matrix to specific OTP + Elixir versions * Run formatter on telemetry and process_propagator * Run formatter over opentelemetry_phoenix --------- Co-authored-by: Tristan Sloughter <t@crashfast.com>
OpentelemetryHTTPoison
OpentelemetryHTTPoison is a opentelemetry-instrumented wrapper around HTTPoison.
Usage
Replace usages of the HTTPoison
module with OpentelemetryHTTPoison
when calling one of the derived request functions provided by HTTPoison
(HTTPoison.get/3
, HTTPoison.get!/3
etc.)
# Before
HTTPoison.get!(url, headers, opts)
# After
OpentelemetryHTTPoison.get!(url, headers, opts)
Configuration
OpentelemetryHTTPoison can be configured through config :opentelemetry_httpoison
. The configurable options are:
-
:ot_attributes
: what default Open Telemetry metadata attributes will be sent per requestIf no value is provided, then no default Open Telemetry metadata attributes will sent per request by default
If a
list
of two elementtuple
s (both elements ofString.t()
) is provided, then these will form the default Open Telemetry metadata attributes sent per requestThe first element of a provided
tuple
is the attribute name, e.g.service.name
, whilst the second element is the attribute value, e.g. "shoppingcart" -
:infer_route
: how thehttp.route
Open Telemetry metadata will be set per requestIf no value is provided then the out of the box, conservative inference provided by
OpentelemetryHTTPoison.URI.infer_route_from_request/1
is used to determine the inferenceIf a function with an arity of 1 (the argument given being the
t:HTTPoison.Request/0
request
) is provided then that function is used to determine the inference
Both of these can be overridden per each call to OpentelemetryHTTPoison functions that wrap OpentelemetryHTTPoison.request/1
, such as OpentelemetryHTTPoison.get/3
, OpentelemetryHTTPoison.get!/3
, OpentelemetryHTTPoison.post/3
etc.
See here for examples
Open Telemetry integration
Additionally, OpentelemetryHTTPoison
provides some options that can be added to each derived function via
the Keyword list
opts
parameter (or the t:HTTPoison.Request/0
Keyword list
options
parameter if calling OpentelemetryHTTPoison.Request/1
directly). These are prefixed with :ot_
.
:ot_span_name
- sets the span name.:ot_attributes
- a list of{name, value}
tuple
attributes that will be added to the span.:ot_resource_route
- sets thehttp.route
attribute, depending on the value provided.
If the value is a string or an function with an arity of 1 (the t:HTTPoison.Request/0
request
) that is used to set the attribute
If :infer
is provided, then the function discussed within the Configuration section is used to set the attribute
If the atom :ignore
is provided then the http.route
attribute is ignored entirely
It is highly recommended to supply the :ot_resource_route
explicitly as either a string or a function with an arity of 1 (the t:HTTPoison.Request/0
request
)
Examples
In the below examples, OpentelemetryHTTPoison.get!/3
is used for the sake of simplicity but other functions derived from OpentelemetryHTTPoison.request/1
can be used
config :opentelemetry_httpoison,
ot_attributes: [{"service.name", "users"}]
OpentelemetryHTTPoison.get!(
"https://www.example.com/user/list",
[],
ot_span_name: "list example users",
ot_attributes: [{"example.language", "en"}],
ot_resource_route: :infer
)
In the example above:
- OpentelemetryHTTPoison is configured with
{"service.name", "users"}
as the value for the:ot_attributes
option :infer
is passed as the value for the:ot_resource_route
Keyword list
option
Given the above, the service.name
attribute will be set to "users" and the http.route
attribute will be inferred as /user/:subpath
OpentelemetryHTTPoison.get!(
"https://www.example.com/user/list",
[],
ot_span_name: "list example users",
ot_attributes: [{"example.language", "en"}],
ot_resource_route: :infer
)
In the example above:
:infer
is passed as the value for:ot_resource_route
Keyword list
option
Given the above, the http.route
attribute will be inferred as /user/:subpath
config :opentelemetry_httpoison,
infer_route: fn
%HTTPoison.Request{} = request -> URI.parse(request.url).path
end
OpentelemetryHTTPoison.get!(
"https://www.example.com/user/list",
[],
ot_resource_route: :infer
)
In the example above:
- OpentelemetryHTTPoison is configured with the
:infer_route
option set to a function which takes a%HTTPoison.Request/0
argument, returning the path of the request URL :infer
is passed as the value for:ot_resource_route
Keyword list
option
Given the above, the http.route
attribute will be inferred as /user/list
OpentelemetryHTTPoison.get!(
"https://www.example.com/user/list",
[],
ot_resource_route: "my secret path"
)
In the example above:
"my secret path"
is passed as the value for:ot_resource_route
Keyword list
option
Given the above, the http.route
attribute will be set as my secret path
OpentelemetryHTTPoison.get!(
"https://www.example.com/user/list",
[],
ot_resource_route: :ignore
)
In the example above:
:ignore
is passed as the value for:ot_resource_route
Keyword list
option
Given the above, the http.route
attribute will not be set to any value
How it works
OpentelemetryHTTPoison, when executing an HTTP request to an external service, creates an OpenTelemetry span, injects
the trace context propagation headers in the request headers, and
ends the span once the response is received.
It automatically sets some of the HTTP span attributes like http.status
etc,
based on the request and response data.
OpentelemetryHTTPoison by itself is not particularly useful: it becomes useful when used in conjunction with a "server-side" opentelemetry-instrumented library, e.g. opentelemetry_plug. These do the opposite work: they take the trace context information from the request headers, and they create a SERVER span which becomes the currently active span.
Using the two libraries together, it's possible to propagate trace information across several microservices and through HTTP "jumps".
Keep in mind
-
The Erlang opentelemetry SDK stores the currently active span in a
pdict
, a per-process dict. If OpentelemetryHTTPoison is called from a different process than the one that initially handled the request and created the "server-side" span, OpentelemetryHTTPoison won't find a parent span and will create a new root client span, losing the trace context. In this case, your only option to correctly propagate the trace context is to manually pass around the parent span, and pass it to OpentelemetryHTTPoison when doing the HTTP client request. -
If the request fails due to nxdomain, the
process_response_status_code
hook is not called and therefore the span is not ended.
What's missing
- Set SpanKind to client
- Support for explicit parent span
- Support for fixed span attributes
- Lots of other stuff...