package progress
User-definable progress bars
Install
Dune Dependency
Authors
Maintainers
Sources
progress-0.3.0.tbz
sha256=3dfd00bd4def773239159b17781d02fdbfd8ea191801681a94aa0a5be1d06b7c
sha512=fd64ff8a819b2db2460c06b7fbd5663e2a1941f9e2d4e9b921a3d5f24509fe3be543521fbe1bb6baedad9f62b579aae933efac3903db03a27385233f5461f09c
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 99 100
(*———————————————————————————————————————————————————————————————————————————— 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 -> a = fun t -> let (module Integer) = t.elt in if is_empty t then Integer.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 Integer.zero else let est = Integer.to_float sum /. interval in Integer.of_float est (*———————————————————————————————————————————————————————————————————————————— 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)"
>