package progress
User-definable progress bars
Install
Dune Dependency
Authors
Maintainers
Sources
progress-0.4.0.tbz
sha256=8be449553379bb2dc5e8b79805c80447690a03dca3e9aee959fecf46d8278fb7
sha512=841383e8aa7d6bd802ced5eb7feae01bd507b2914eb45e8a559140677f83d5b8ec614f1d0bc54421021b5254a1edd78dd8a2506b2dfb264af72448d76bd03ac5
doc/src/progress.engine/flow_meter.ml.html
Source file flow_meter.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
(*———————————————————————————————————————————————————————————————————————————— Copyright (c) 2020–2021 Craig Ferguson <me@craigfe.io> Distributed under the MIT license. See terms at the end of this file. ————————————————————————————————————————————————————————————————————————————*) open! Import type 'a t = { data : 'a array ; timestamps : Mtime.t array ; max_length : int ; mutable most_recently_added : int ; mutable length : int ; get_time : unit -> Mtime.t ; elt : (module Integer.S with type t = 'a) } let create (type a) ~clock:get_time ~size ~elt : a t = if size <= 0 then Fmt.invalid_arg "Flow_meter.create: non-positive size %d" size; (* We need [n + 1] timestamp samples to integrate over [n] values. *) let max_length = size + 1 in let start_time = get_time () in { get_time ; data = Array.make max_length (Obj.magic None) ; timestamps = Array.make max_length start_time ; max_length ; most_recently_added = -1 ; length = 0 ; elt } let is_empty t = t.most_recently_added = -1 let push t ~key ~data = t.data.(key) <- data; t.timestamps.(key) <- t.get_time () let record t data = if t.length = t.max_length then ( (* Buffer is full. Overwrite the oldest value. *) let next = (t.most_recently_added + 1) mod t.length in push t ~key:next ~data; t.most_recently_added <- next) else ( (* Increase the buffer size *) push t ~key:t.length ~data; t.most_recently_added <- t.length; t.length <- succ t.length) let oldest_index t = if t.length = t.max_length then (t.most_recently_added + 1) mod t.length else 0 let fold = let rec aux data f acc = function | -1 -> acc | n -> aux data f (f acc data.(n)) (n - 1) in fun t ~f ~init -> aux t.data f init (t.length - 1) let per_second : type a. a t -> float = fun t -> let (module Integer) = t.elt in if is_empty t then Float.zero else (* Sum all values in the window {i except the first one} and divide by the time interval. We can think of the first value as representing work done just {i before} the time interval starts, so using a half-open sample correctly avoids over-reporting the flow-rate. *) let oldest_index = oldest_index t in let sum = Integer.sub (fold t ~f:Integer.add ~init:Integer.zero) t.data.(oldest_index) in let interval = let start_time = t.timestamps.(oldest_index) in let end_time = t.timestamps.(t.most_recently_added) in Mtime.span_to_s (Mtime.span start_time end_time) in if Float.compare interval Float.epsilon < 0 then Float.zero else Integer.to_float sum /. interval (*———————————————————————————————————————————————————————————————————————————— Copyright (c) 2020–2021 Craig Ferguson <me@craigfe.io> Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ————————————————————————————————————————————————————————————————————————————*)
sectionYPositions = computeSectionYPositions($el), 10)"
x-init="setTimeout(() => sectionYPositions = computeSectionYPositions($el), 10)"
>