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
use crate::bounds::Bounds;
use crate::duration::Duration;
use crate::groupby::TimeUnit;
#[derive(Copy, Clone)]
pub struct Window {
every: Duration,
period: Duration,
offset: Duration,
}
impl Window {
pub fn new(every: Duration, period: Duration, offset: Duration) -> Self {
Self {
every,
period,
offset,
}
}
#[inline]
pub fn truncate_ns(&self, t: i64) -> i64 {
let t = self.every.truncate_ns(t);
self.offset.add_ns(t)
}
#[inline]
pub fn truncate_no_offset_ns(&self, t: i64) -> i64 {
self.every.truncate_ns(t)
}
#[inline]
pub fn truncate_ms(&self, t: i64) -> i64 {
let t = self.every.truncate_ms(t);
self.offset.add_ms(t)
}
#[inline]
pub fn truncate_no_offset_ms(&self, t: i64) -> i64 {
self.every.truncate_ms(t)
}
pub fn get_earliest_bounds_ns(&self, t: i64) -> Bounds {
let start = self.truncate_ns(t);
let stop = self.period.add_ns(start);
Bounds::new(start, stop)
}
pub fn get_earliest_bounds_ms(&self, t: i64) -> Bounds {
let start = self.truncate_ms(t);
let stop = self.period.add_ms(start);
Bounds::new(start, stop)
}
pub(crate) fn estimate_overlapping_bounds_ns(&self, boundary: Bounds) -> usize {
(boundary.duration() / self.every.duration_ns()
+ self.period.duration_ns() / self.every.duration_ns()) as usize
}
pub(crate) fn estimate_overlapping_bounds_ms(&self, boundary: Bounds) -> usize {
(boundary.duration() / self.every.duration_ms()
+ self.period.duration_ms() / self.every.duration_ms()) as usize
}
pub fn get_overlapping_bounds_iter(&self, boundary: Bounds, tu: TimeUnit) -> BoundsIter {
BoundsIter::new(*self, boundary, tu)
}
}
pub struct BoundsIter {
window: Window,
boundary: Bounds,
bi: Bounds,
tu: TimeUnit,
}
impl BoundsIter {
fn new(window: Window, boundary: Bounds, tu: TimeUnit) -> Self {
let bi = match tu {
TimeUnit::Nanoseconds => window.get_earliest_bounds_ns(boundary.start),
TimeUnit::Milliseconds => window.get_earliest_bounds_ms(boundary.start),
};
Self {
window,
boundary,
bi,
tu,
}
}
}
impl Iterator for BoundsIter {
type Item = Bounds;
fn next(&mut self) -> Option<Self::Item> {
if self.bi.start < self.boundary.stop {
let out = self.bi;
match self.tu {
TimeUnit::Nanoseconds => {
self.bi.start = self.window.every.add_ns(self.bi.start);
self.bi.stop = self.window.every.add_ns(self.bi.stop);
}
TimeUnit::Milliseconds => {
self.bi.start = self.window.every.add_ms(self.bi.start);
self.bi.stop = self.window.every.add_ms(self.bi.stop);
}
}
Some(out)
} else {
None
}
}
}