Source file dune_action_trace.ml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
module List = ListLabels

module Private = struct
  let trace_dir_env_var = "DUNE_ACTION_TRACE_DIR"
end

module Event = struct
  type t = Csexp.t

  module Arg = struct
    let string s = Csexp.Atom s
    let int x = string (string_of_int x)
    let list xs = Csexp.List xs
    let record xs = List.map xs ~f:(fun (k, v) -> list [ string k; v ])
    let time_ns ts = int ts
    let span_ns span = int span
  end

  let base ~name cat : Csexp.t list = [ Atom cat; Atom name ]

  let instant ?(args = []) ~category ~name ~time_in_nanoseconds () =
    Csexp.List
      (base ~name category @ [ Arg.time_ns time_in_nanoseconds ] @ Arg.record args)
  ;;

  let span ?(args = []) ~category ~name ~start_in_nanoseconds ~duration_in_nanoseconds () =
    Csexp.List
      (base ~name category
       @ [ Csexp.List
             [ Arg.time_ns start_in_nanoseconds; Arg.span_ns duration_in_nanoseconds ]
         ]
       @ Arg.record args)
  ;;
end

module Sys = struct
  let[@warning "-32"] mkdir _ = failwith "unsupported"

  include Sys
end

module Context = struct
  type state =
    | Open of out_channel
    | Closed

  type t =
    | Disabled
    | Enabled of state ref

  let create ~name =
    match Sys.getenv_opt Private.trace_dir_env_var with
    | None -> Disabled
    | Some dir ->
      (match
         match Sys.mkdir dir 0o777 with
         | () -> `Ok
         | exception Sys_error _ when Sys.is_directory dir -> `Ok
         | exception _ -> `Failure
       with
       | `Failure -> Disabled
       | `Ok ->
         Enabled (ref (Open (Filename.open_temp_file ~temp_dir:dir name ".csexp" |> snd))))
  ;;

  let is_enabled = function
    | Disabled -> false
    | Enabled x ->
      (match !x with
       | Open _ -> true
       | Closed -> false)
  ;;

  let emit t event =
    match t with
    | Disabled -> ()
    | Enabled state ->
      (match !state with
       | Closed -> failwith "unable to log event because context is closed"
       | Open chan -> Csexp.to_channel chan event)
  ;;

  let close t =
    match t with
    | Disabled -> ()
    | Enabled state ->
      (match !state with
       | Closed -> ()
       | Open chan ->
         state := Closed;
         close_out chan)
  ;;
end