Skip to content

Commit

Permalink
fix: wait for cleanup in cohttp client
Browse files Browse the repository at this point in the history
in `Opentelemetry_client_cohttp_lwt.with_setup` we should now wait for
the cleanup to be done, by sneaking in a `unit Lwt.u` that is only
resolved after the cleanup is done.

close #41
  • Loading branch information
c-cube committed Oct 17, 2024
1 parent e789ecf commit 55977b1
Show file tree
Hide file tree
Showing 2 changed files with 43 additions and 14 deletions.
50 changes: 38 additions & 12 deletions src/client-cohttp-lwt/opentelemetry_client_cohttp_lwt.ml
Original file line number Diff line number Diff line change
Expand Up @@ -258,7 +258,8 @@ end
exceptions inside should be caught, see
https://opentelemetry.io/docs/reference/specification/error-handling/ *)
let mk_emitter ~stop ~(config : Config.t) () : (module EMITTER) =
let mk_emitter ~(after_cleanup : unit Lwt.u option) ~stop ~(config : Config.t)
() : (module EMITTER) =
let open Proto in
let open Lwt.Syntax in
(* local helpers *)
Expand Down Expand Up @@ -448,6 +449,8 @@ let mk_emitter ~stop ~(config : Config.t) () : (module EMITTER) =
Lwt.async (fun () ->
let* () = emit_all_force httpc encoder in
Httpc.cleanup httpc;
(* resolve [after_cleanup], if provided *)
Option.iter (fun prom -> Lwt.wakeup_later prom ()) after_cleanup;
Lwt.return ())
end in
(module M)
Expand All @@ -457,9 +460,13 @@ module Backend
val stop : bool Atomic.t

val config : Config.t

val after_cleanup : unit Lwt.u option
end)
() : Opentelemetry.Collector.BACKEND = struct
include (val mk_emitter ~stop:Arg.stop ~config:Arg.config ())
include
(val mk_emitter ~after_cleanup:Arg.after_cleanup ~stop:Arg.stop
~config:Arg.config ())

open Opentelemetry.Proto
open Opentelemetry.Collector
Expand Down Expand Up @@ -551,7 +558,8 @@ module Backend
}
end

let create_backend ?(stop = Atomic.make false) ?(config = Config.make ()) () =
let create_backend ?after_cleanup ?(stop = Atomic.make false)
?(config = Config.make ()) () =
debug_ := config.debug;

let module B =
Expand All @@ -560,25 +568,43 @@ let create_backend ?(stop = Atomic.make false) ?(config = Config.make ()) () =
let stop = stop

let config = config

let after_cleanup = after_cleanup
end)
()
in
(module B : OT.Collector.BACKEND)

let setup_ ?stop ?config () =
let backend = create_backend ?stop ?config () in
let setup_ ?stop ?config () : (unit -> unit) * unit Lwt.t =
let cleanup_done, cleanup_done_prom = Lwt.wait () in
let backend =
create_backend ~after_cleanup:cleanup_done_prom ?stop ?config ()
in
OT.Collector.set_backend backend;
OT.Collector.remove_backend

OT.Collector.remove_backend, cleanup_done

let setup ?stop ?config ?(enable = true) () =
if enable then (
let cleanup = setup_ ?stop ?config () in
let cleanup, _lwt = setup_ ?stop ?config () in
at_exit cleanup
)

let with_setup ?stop ?(config = Config.make ()) ?(enable = true) () f =
if enable then (
let cleanup = setup_ ?stop ~config () in
Fun.protect ~finally:cleanup f
) else
let with_setup ?stop ?(config = Config.make ()) ?(enable = true) () f : _ Lwt.t
=
if enable then
let open Lwt.Syntax in
let cleanup, cleanup_done = setup_ ?stop ~config () in

Lwt.catch
(fun () ->
let* res = f () in
cleanup ();
let+ () = cleanup_done in
res)
(fun exn ->
cleanup ();
let* () = cleanup_done in
Lwt.reraise exn)
else
f ()
7 changes: 5 additions & 2 deletions src/client-cohttp-lwt/opentelemetry_client_cohttp_lwt.mli
Original file line number Diff line number Diff line change
Expand Up @@ -13,10 +13,13 @@ val set_headers : (string * string) list -> unit
module Config = Config

val create_backend :
?after_cleanup:unit Lwt.u ->
?stop:bool Atomic.t ->
?config:Config.t ->
unit ->
(module Opentelemetry.Collector.BACKEND)
(** Create a new backend using lwt and cohttp
@param after_cleanup if provided, this is resolved into [()] after cleanup is done (since NEXT_RELEASE) *)

val setup :
?stop:bool Atomic.t -> ?config:Config.t -> ?enable:bool -> unit -> unit
Expand All @@ -34,8 +37,8 @@ val with_setup :
?config:Config.t ->
?enable:bool ->
unit ->
(unit -> 'a) ->
'a
(unit -> 'a Lwt.t) ->
'a Lwt.t
(** [with_setup () f] is like [setup(); f()] but takes care of cleaning up
after [f()] returns
See {!setup} for more details. *)

0 comments on commit 55977b1

Please sign in to comment.