Source file line.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
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
include Line_intf
module Primitives = Line_primitives
open! Import
(** [Line] is a higher-level wrapper around [Segment] that makes some
    simplifying assumptions about progress bar rendering:
    - the reported value has a monoid instance, used for initialisation and
      accumulation.
    - the line is wrapped inside a single box.
    It contains its own notion of "accumulated" line segments, and ensures that
    this works with conditional rendering of segments (e.g. when rendering with
    a minimum interval). *)
module Acc = struct
  type 'a t =
    { mutable latest : 'a
    ; mutable accumulator : 'a
    ; mutable pending : 'a
    ; render_start : Mtime.t
    ; flow_meter : 'a Flow_meter.t
    }
  let wrap :
      type a.
         elt:(module Integer.S with type t = a)
      -> clock:(unit -> Mtime.t)
      -> should_update:(unit -> bool)
      -> a t Primitives.t
      -> a Primitives.t =
   fun ~elt:(module Integer) ~clock ~should_update inner ->
    Primitives.stateful (fun () ->
        let flow_meter =
          Flow_meter.create ~clock ~size:32 ~elt:(module Integer)
        in
        let render_start = clock () in
        let state =
          { latest = Integer.zero
          ; accumulator = Integer.zero
          ; pending = Integer.zero
          ; render_start
          ; flow_meter
          }
        in
        Primitives.contramap ~f:(fun a ->
            Flow_meter.record state.flow_meter a;
            state.pending <- Integer.add a state.pending)
        @@ Primitives.conditional (fun _ -> should_update ())
        
        @@ Primitives.on_finalise ()
        @@ Primitives.contramap ~f:(fun () ->
               let to_record = state.pending in
               state.accumulator <- Integer.add to_record state.accumulator;
               state.latest <- to_record;
               state.pending <- Integer.zero;
               state)
        @@ inner)
  let accumulator t = t.accumulator
  let flow_meter t = t.flow_meter
end
module Timer = struct
  type 'a t = { mutable render_latest : Mtime.t }
  let should_update ~interval ~clock t =
    match interval with
    | None -> Staged.inj (fun () -> true)
    | Some interval ->
        Staged.inj (fun () ->
            let now = clock () in
            match
              Mtime.Span.compare (Mtime.span t.render_latest now) interval >= 0
            with
            | false -> false
            | true ->
                t.render_latest <- now;
                true)
end
type 'a t =
  | Noop
  | Primitive of 'a Primitives.t
  | Basic of 'a Primitives.t
  | Map of ('a Primitives.t -> 'a Primitives.t) * 'a t
  | List of 'a t list
  | Contramap : ('b t * ('a -> 'b)) -> 'a t
  | Pair : 'a t * unit t * 'b t -> ('a * 'b) t
  | Acc of
      { segment : 'a Acc.t Primitives.t
      ; elt : (module Integer.S with type t = 'a)
      }
module Integer_independent (Platform : Platform.S) = struct
  open struct
    module Clock = Platform.Clock
  end
  let noop () = Noop
  let const s =
    let len = String.length s and width = Terminal.guess_printed_width s in
    let segment =
      Primitives.theta ~width (fun buf _ ->
          Line_buffer.add_substring buf s ~off:0 ~len)
    in
    Basic segment
  let spacer n = const (String.make n ' ')
  
  let str_formatter_buf, str_formatter =
    let buf = Buffer.create 0 in
    let ppf = Format.formatter_of_buffer buf in
    Fmt.set_style_renderer ppf `Ansi_tty;
    (buf, ppf)
  let constf fmt =
    Fmt.kpf
      (fun ppf ->
        Format.pp_print_flush ppf ();
        let str = Buffer.contents str_formatter_buf in
        Buffer.clear str_formatter_buf;
        const str)
      str_formatter fmt
  let pair ?(sep = noop ()) a b = Pair (a, sep, b)
  let list ?(sep = const " ") xs =
    let xs =
      ListLabels.filter_map xs ~f:(function Noop -> None | x -> Some x)
    in
    List (List.intersperse ~sep xs)
  let ( ++ ) a b = List [ a; b ]
  let parens t = const "(" ++ t ++ const ")"
  let brackets t = const "[" ++ t ++ const "]"
  let braces t = const "{" ++ t ++ const "}"
  let using f x = Contramap (x, f)
  let string =
    let segment =
      Primitives.alpha_unsized ~initial:(`Val "") (fun ~width buf _ s ->
          let output_len = width () - 1  in
          if output_len <= 0 then 0
          else
            let pp =
              Staged.prj
              @@ Printer.Internals.to_line_printer
                   (Printer.string ~width:output_len)
            in
            pp buf s;
            output_len)
    in
    Basic segment
  let lpad sz t = Map (Primitives.box_fixed ~pad:`left sz, t)
  let rpad sz t = Map (Primitives.box_fixed ~pad:`right sz, t)
  
  let with_color_opt color buf f =
    match color with
    | None -> f ()
    | Some s ->
        Line_buffer.add_string buf Terminal.Style.(code (fg s));
        let a = f () in
        Line_buffer.add_string buf Terminal.Style.(code none);
        a
  module Modulo_counter : sig
    type t
    val create : int -> t
    val latest : t -> int
    val tick : t -> int
  end = struct
    type t = { modulus : int; mutable latest : int }
    let create modulus = { modulus; latest = 0 }
    let latest t = t.latest
    let tick t =
      t.latest <- succ t.latest mod t.modulus;
      t.latest
  end
  let debounce interval s =
    Primitives.stateful (fun () ->
        let latest = ref (Clock.now ()) in
        let should_update () =
          let now = Clock.now () in
          match Mtime.Span.compare (Mtime.span !latest now) interval >= 0 with
          | false -> false
          | true ->
              latest := now;
              true
        in
        Primitives.conditional (fun _ -> should_update ()) s)
  module Spinner = struct
    type t = { frames : string array; final_frame : string option; width : int }
    let v ~frames ~final_frame ~width = { frames; final_frame; width }
    let default =
      v ~final_frame:(Some "✔️") ~width:1
        ~frames:[| "⠋"; "⠙"; "⠹"; "⠸"; "⠼"; "⠴"; "⠦"; "⠧"; "⠇"; "⠏" |]
    let stage_count t = Array.length t.frames
  end
  let spinner ?frames ?color ?(min_interval = Some (Duration.of_int_ms 80)) () =
    let spinner =
      match frames with
      | None -> Spinner.default
      | Some [] -> Fmt.invalid_arg "spinner must have at least one stage"
      | Some (x :: xs as frames) ->
          let width = Terminal.guess_printed_width x in
          ListLabels.iteri xs ~f:(fun i x ->
              let width' = Terminal.guess_printed_width x in
              if width <> width' then
                Fmt.invalid_arg
                  "Spinner frames must have the same UTF-8 length. found %d \
                   (at index 0) and %d (at index %d)"
                  (i + 1) width width');
          Spinner.v ~frames:(Array.of_list frames) ~final_frame:None ~width
    in
    let apply_debounce = Option.fold min_interval ~none:Fun.id ~some:debounce in
    Basic
      (Primitives.stateful (fun () ->
           let counter = Modulo_counter.create (Spinner.stage_count spinner) in
           apply_debounce
           @@ Primitives.theta ~width:spinner.Spinner.width (fun buf -> function
                | `finish ->
                    let final_frame =
                      match spinner.final_frame with
                      | None -> spinner.frames.(Modulo_counter.tick counter)
                      | Some x -> x
                    in
                    with_color_opt color buf (fun () ->
                        Line_buffer.add_string buf final_frame)
                | (`report | `tick | `rerender) as e ->
                    let tick =
                      match e with
                      | `report | `tick -> Modulo_counter.tick counter
                      | `rerender -> Modulo_counter.latest counter
                    in
                    let frame = spinner.Spinner.frames.(tick) in
                    with_color_opt color buf (fun () ->
                        Line_buffer.add_string buf frame))))
end
module Bar_style = struct
  type t =
    { delimiters : (string * string) option
    ; blank_space : string
    ; full_space : string
    ; in_progress_stages : string array
    ; color : Terminal.Color.t option
    ; color_empty : Terminal.Color.t option
    ; total_delimiter_width : int
    ; segment_width : int
    }
  let ascii =
    { delimiters = Some ("[", "]")
    ; blank_space = "-"
    ; full_space = "#"
    ; in_progress_stages = [||]
    ; color = None
    ; color_empty = None
    ; total_delimiter_width = 2
    ; segment_width = 1
    }
  let utf8 =
    { delimiters = Some ("│", "│")
    ; blank_space = " "
    ; full_space = "█"
    ; in_progress_stages = [| " "; "▏"; "▎"; "▍"; "▌"; "▋"; "▊"; "▉" |]
    ; color = None
    ; color_empty = None
    ; total_delimiter_width = 2
    ; segment_width = 1
    }
  let parse_stages ctx = function
    | [] -> Fmt.invalid_arg "%s: empty list of bar stages supplied" ctx
    | full_space :: xs ->
        let segment_width = Terminal.guess_printed_width full_space in
        if segment_width = 0 then
          Fmt.invalid_arg
            "%s: supplied stage '%s' has estimated printed width of 0" ctx
            full_space;
        let in_progress_stages, blank_space =
          match List.rev xs with
          | [] -> ([||], String.make segment_width ' ')
          | blank_space :: xs -> (Array.of_list xs, blank_space)
        in
        (full_space, in_progress_stages, blank_space, segment_width)
  let guess_delims_width = function
    | None -> 0
    | Some (l, r) -> Terminal.(guess_printed_width l + guess_printed_width r)
  let v ?delims ?color ?color_empty stages =
    let full_space, in_progress_stages, blank_space, segment_width =
      parse_stages "Bar_styles.v" stages
    in
    { delimiters = delims
    ; blank_space
    ; full_space
    ; in_progress_stages
    ; color
    ; color_empty
    ; segment_width
    ; total_delimiter_width = guess_delims_width delims
    }
  let with_color color t = { t with color = Some color }
  let with_empty_color color_empty t = { t with color_empty = Some color_empty }
  let with_delims delimiters t =
    { t with delimiters; total_delimiter_width = guess_delims_width delimiters }
  let with_stages stages t =
    let full_space, in_progress_stages, blank_space, segment_width =
      parse_stages "Bar_styles.with_stages" stages
    in
    { t with full_space; blank_space; in_progress_stages; segment_width }
end
module Make (Platform : Platform.S) = struct
  open struct
    module Clock = Platform.Clock
  end
  module Integer_independent = Integer_independent (Platform)
  include Integer_independent
  module Internals = struct
    module Line_buffer = Line_buffer
    include Primitives
    let box_winsize ?max ?(fallback = 80) s =
      let get_width () =
        let real_width =
          Option.value ~default:fallback (Platform.Terminal_width.get ())
        in
        match max with None -> real_width | Some m -> min m real_width
      in
      box_dynamic get_width s
    let to_line t = Primitive t
  end
  let to_primitive : type a. Config.t -> a t -> a Primitives.t =
    let rec inner : type a. a t -> (unit -> bool) -> a Primitives.t = function
      | Noop -> fun _ -> Primitives.noop ()
      | Primitive x -> fun _ -> x
      | Pair (a, sep, b) ->
          let a = inner a in
          let sep = inner sep in
          let b = inner b in
          fun should_update ->
            Primitives.pair ~sep:(sep should_update) (a should_update)
              (b should_update)
      | Contramap (x, f) ->
          let x = inner x in
          fun y -> Primitives.contramap ~f (x y)
      | Map (f, x) -> fun a -> f (inner x a)
      | List xs ->
          let xs = List.map xs ~f:inner in
          fun should_update ->
            Primitives.array
              (List.map xs ~f:(fun f -> f should_update) |> Array.of_list)
      | Basic segment ->
          fun should_update ->
            Primitives.conditional (fun _ -> should_update ()) @@ segment
      | Acc { segment; elt = (module Integer) } ->
          fun should_update ->
            Acc.wrap ~elt:(module Integer) ~clock:Clock.now ~should_update
            @@ segment
    in
    fun (config : Config.t) -> function
      | Primitive x -> x
      | t ->
          let inner = inner t in
          let segment =
            Primitives.stateful (fun () ->
                let should_update =
                  let state = { Timer.render_latest = Clock.now () } in
                  Staged.prj
                    (Timer.should_update ~clock:Clock.now
                       ~interval:config.min_interval state)
                in
                let x = ref true in
                Primitives.contramap ~f:(fun a ->
                    x := should_update ();
                    a)
                @@ Internals.box_winsize ?max:config.max_width
                @@ inner (fun () -> !x))
          in
          segment
  
  module Integer_dependent = struct
    module type S =
      Integer_dependent
        with type 'a t := 'a t
         and type color := Terminal.Color.t
         and type duration := Duration.t
         and type 'a printer := 'a Printer.t
         and type bar_style := Bar_style.t
    module type Ext =
      DSL
        with type 'a t := 'a t
         and type color := Terminal.Color.t
         and type duration := Duration.t
         and type 'a printer := 'a Printer.t
         and type Bar_style.t := Bar_style.t
    module Make_ext (Integer : Integer.S) = struct
      let acc segment = Acc { segment; elt = (module Integer) }
      let of_printer ?init printer =
        let pp = Staged.prj @@ Printer.Internals.to_line_printer printer in
        let width = Printer.print_width printer in
        let initial =
          match init with
          | Some v -> `Val v
          | None ->
              `Theta
                (fun buf ->
                  for _ = 1 to width do
                    Line_buffer.add_char buf ' '
                  done)
        in
        Basic (Primitives.alpha ~width ~initial (fun buf _ x -> pp buf x))
      let count_pp printer =
        let pp = Staged.prj @@ Printer.Internals.to_line_printer printer in
        acc
        @@ Primitives.contramap ~f:Acc.accumulator
        @@ Primitives.alpha ~width:(Printer.print_width printer)
             ~initial:(`Val Integer.zero) (fun buf _ x -> pp buf x)
      let bytes = count_pp (Units.Bytes.generic (module Integer))
      let percentage_of accumulator =
        let printer =
          Printer.using Units.Percentage.of_float ~f:(fun x ->
              Integer.to_float x /. Integer.to_float accumulator)
        in
        count_pp printer
      let sum ?pp ~width () =
        let pp =
          match pp with
          | None -> Printer.Internals.integer ~width (module Integer)
          | Some x -> x
        in
        let pp = Staged.prj (Printer.Internals.to_line_printer pp) in
        acc
        @@ Primitives.contramap ~f:Acc.accumulator
        @@ Primitives.alpha ~initial:(`Val Integer.zero) ~width (fun buf _ x ->
               pp buf x)
      let count_to ?pp ?(sep = const "/") total =
        let total = Integer.to_string total in
        let width =
          match pp with
          | Some pp -> Printer.print_width pp
          | None -> String.length total
        in
        List [ sum ~width (); using (fun _ -> ()) sep; const total ]
      let ticker_to ?(sep = const "/") total =
        let total = Integer.to_string total in
        let width = String.length total in
        let pp =
          Staged.prj
          @@ Printer.Internals.to_line_printer
          @@ Printer.Internals.integer ~width (module Integer)
        in
        let segment =
          Primitives.alpha ~width ~initial:(`Val Integer.zero) (fun buf _ x ->
              pp buf x)
        in
        List
          [ Contramap
              ( Acc
                  { segment = Primitives.contramap ~f:Acc.accumulator segment
                  ; elt = (module Integer)
                  }
              , fun _ -> Integer.one )
          ; using (fun _ -> ()) sep
          ; const total
          ]
      
      module Bar_style = Bar_style
      let bar (spec : Bar_style.t) width proportion buf =
        let final_stage = Array.length spec.in_progress_stages in
        let width = width () in
        let bar_segments =
          (width - spec.total_delimiter_width) / spec.segment_width
        in
        let squaresf = Float.of_int bar_segments *. proportion in
        let squares = Float.to_int squaresf in
        let filled = min squares bar_segments in
        let not_filled =
          bar_segments - filled - if final_stage = 0 then 0 else 1
        in
        Option.iter (fun (x, _) -> Line_buffer.add_string buf x) spec.delimiters;
        with_color_opt spec.color buf (fun () ->
            for _ = 1 to filled do
              Line_buffer.add_string buf spec.full_space
            done);
        let () =
          if filled <> bar_segments then (
            let chunks = Float.to_int (squaresf *. Float.of_int final_stage) in
            let index = chunks - (filled * final_stage) in
            if index >= 0 && index < final_stage then
              with_color_opt spec.color buf (fun () ->
                  Line_buffer.add_string buf spec.in_progress_stages.(index));
            with_color_opt spec.color_empty buf (fun () ->
                for _ = 1 to not_filled do
                  Line_buffer.add_string buf spec.blank_space
                done))
        in
        Option.iter (fun (_, x) -> Line_buffer.add_string buf x) spec.delimiters;
        width
      let with_prop f v t = match v with None -> t | Some v -> f v t
      let bar ~style ~color =
        let style =
          match style with
          | `ASCII -> Bar_style.ascii
          | `UTF8 -> Bar_style.utf8
          | `Custom style -> style
        in
        bar (style |> with_prop Bar_style.with_color color)
      let bar ?(style = `ASCII) ?color ?(width = `Expand) ?(data = `Sum) total =
        let proportion x = Integer.to_float x /. Integer.to_float total in
        let proportion_segment =
          match width with
          | `Fixed width ->
              if width < 3 then failwith "Not enough space for a progress bar";
              Primitives.alpha ~width ~initial:(`Val 0.) (fun buf _ x ->
                  ignore (bar ~style ~color (fun _ -> width) x buf : int))
          | `Expand ->
              Primitives.alpha_unsized ~initial:(`Val 0.) (fun ~width ppf _ x ->
                  bar ~style ~color width x ppf)
        in
        match data with
        | `Latest ->
            Basic (Primitives.contramap proportion_segment ~f:proportion)
        | `Sum ->
            acc
              (Primitives.contramap proportion_segment
                 ~f:(Acc.accumulator >> proportion))
      let rate pp_val =
        let pp_rate =
          let pp_val = Staged.prj (Printer.Internals.to_line_printer pp_val) in
          fun buf _ x ->
            pp_val buf x;
            Line_buffer.add_string buf "/s"
        in
        let width = Printer.print_width pp_val + 2 in
        acc
        @@ Primitives.contramap
             ~f:(Acc.flow_meter >> Flow_meter.per_second >> Integer.to_float)
        @@ Primitives.alpha ~width ~initial:(`Val 0.) pp_rate
      let bytes_per_sec = rate Units.Bytes.of_float
      let eta ?(pp = Units.Duration.mm_ss) total =
        let span_segment =
          let printer =
            let pp = Staged.prj (Printer.Internals.to_line_printer pp) in
            fun ppf event x ->
              match event with
              | `finish -> pp ppf Mtime.Span.max_span 
              | `report | `rerender
              | `tick
                
                ->
                  pp ppf x
          in
          let width = Printer.print_width Units.Duration.mm_ss in
          let initial = `Val Mtime.Span.max_span in
          Primitives.alpha ~width ~initial printer
        in
        acc
        @@ Primitives.contramap ~f:(fun acc ->
               let per_second = Flow_meter.per_second (Acc.flow_meter acc) in
               let acc = Acc.accumulator acc in
               if Integer.(equal zero) per_second then Mtime.Span.max_span
               else
                 let todo = Integer.(to_float (sub total acc)) in
                 if Float.(todo <= 0.) then Mtime.Span.zero
                 else
                   Mtime.Span.of_uint64_ns
                     (Int64.of_float
                        (todo /. Integer.to_float per_second *. 1_000_000_000.)))
        @@ span_segment
      let elapsed ?(pp = Units.Duration.mm_ss) () =
        let print_time = Staged.prj (Printer.Internals.to_line_printer pp) in
        let width = Printer.print_width pp in
        let segment =
          Primitives.stateful (fun () ->
              let elapsed = Clock.counter () in
              let latest = ref Mtime.Span.zero in
              let finished = ref false in
              let pp buf e =
                (match e with
                | `tick | `report -> latest := Clock.count elapsed
                | `finish when not !finished ->
                    latest := Clock.count elapsed;
                    finished := true
                | `rerender | `finish -> ());
                print_time buf !latest
              in
              Primitives.theta ~width pp)
        in
        Basic segment
      include Integer_independent
    end
    module Make = Make_ext
  end
  include Integer_dependent.Make (Integer.Int)
  module Using_int32 = Integer_dependent.Make_ext (Integer.Int32)
  module Using_int63 = Integer_dependent.Make_ext (Integer.Int63)
  module Using_int64 = Integer_dependent.Make_ext (Integer.Int64)
  module Using_float = Integer_dependent.Make_ext (Integer.Float)
end