twilight_gateway/
latency.rs1use std::time::{Duration, Instant};
4
5#[derive(Clone, Debug)]
16pub struct Latency {
17 latency_sum: Duration,
19 periods: u32,
21 received: Option<Instant>,
23 recent: [Duration; Self::RECENT_LEN],
25 sent: Option<Instant>,
27}
28
29impl Latency {
30 const RECENT_LEN: usize = 5;
32
33 pub(crate) const fn new() -> Self {
35 Self {
36 latency_sum: Duration::ZERO,
37 periods: 0,
38 received: None,
39 recent: [Duration::MAX; Self::RECENT_LEN],
40 sent: None,
41 }
42 }
43
44 pub const fn average(&self) -> Option<Duration> {
51 self.latency_sum.checked_div(self.periods)
52 }
53
54 pub const fn periods(&self) -> u32 {
56 self.periods
57 }
58
59 pub fn recent(&self) -> &[Duration] {
61 let maybe_zero_idx = self
65 .recent
66 .iter()
67 .position(|duration| *duration == Duration::MAX);
68
69 &self.recent[0..maybe_zero_idx.unwrap_or(Self::RECENT_LEN)]
70 }
71
72 pub const fn received(&self) -> Option<Instant> {
74 self.received
75 }
76
77 pub const fn sent(&self) -> Option<Instant> {
79 self.sent
80 }
81
82 #[track_caller]
94 pub(crate) fn record_received(&mut self) {
95 debug_assert!(self.received.is_none(), "period completed multiple times");
96
97 let now = Instant::now();
98 let period_latency = now - self.sent.expect("period has not begun");
99 self.received = Some(now);
100 self.periods += 1;
101
102 self.latency_sum += period_latency;
103 self.recent.copy_within(..Self::RECENT_LEN - 1, 1);
104 self.recent[0] = period_latency;
105 }
106
107 pub(crate) fn record_sent(&mut self) {
113 self.received = None;
114 self.sent = Some(Instant::now());
115 }
116}
117
118#[cfg(test)]
119mod tests {
120 use super::Latency;
121 use static_assertions::assert_impl_all;
122 use std::{fmt::Debug, time::Duration};
123
124 assert_impl_all!(Latency: Clone, Debug, Send, Sync);
125
126 const fn default_latency() -> Latency {
127 Latency {
128 latency_sum: Duration::from_millis(510),
129 periods: 17,
130 received: None,
131 recent: [
132 Duration::from_millis(20),
133 Duration::from_millis(25),
134 Duration::from_millis(30),
135 Duration::from_millis(35),
136 Duration::from_millis(40),
137 ],
138 sent: None,
139 }
140 }
141
142 #[test]
143 fn public_api() {
144 let latency = default_latency();
145 assert_eq!(latency.average(), Some(Duration::from_millis(30)));
146 assert_eq!(latency.periods(), 17);
147 assert!(latency.received().is_none());
148 assert!(latency.sent().is_none());
149
150 assert_eq!(latency.recent.len(), Latency::RECENT_LEN);
151 let mut iter = latency.recent().iter();
152 assert_eq!(iter.next(), Some(&Duration::from_millis(20)));
153 assert_eq!(iter.next_back(), Some(&Duration::from_millis(40)));
154 assert_eq!(iter.next(), Some(&Duration::from_millis(25)));
155 assert_eq!(iter.next(), Some(&Duration::from_millis(30)));
156 assert_eq!(iter.next_back(), Some(&Duration::from_millis(35)));
157 assert!(iter.next().is_none());
158 assert!(iter.next_back().is_none());
159 }
160
161 #[test]
164 fn recent() {
165 let no_recents = Latency {
168 latency_sum: Duration::ZERO,
169 periods: 0,
170 received: None,
171 recent: [Duration::MAX; Latency::RECENT_LEN],
172 sent: None,
173 };
174 assert!(no_recents.recent().is_empty());
175
176 let partial = Latency {
179 recent: [
180 Duration::from_millis(40),
181 Duration::from_millis(50),
182 Duration::MAX,
183 Duration::MAX,
184 Duration::MAX,
185 ],
186 ..no_recents
187 };
188 assert_eq!(
189 [Duration::from_millis(40), Duration::from_millis(50)],
190 partial.recent()
191 );
192
193 let full = Latency {
196 recent: [
197 Duration::from_millis(40),
198 Duration::from_millis(50),
199 Duration::from_millis(60),
200 Duration::from_millis(70),
201 Duration::from_millis(60),
202 ],
203 ..no_recents
204 };
205 assert_eq!(
206 [
207 Duration::from_millis(40),
208 Duration::from_millis(50),
209 Duration::from_millis(60),
210 Duration::from_millis(70),
211 Duration::from_millis(60),
212 ],
213 full.recent()
214 );
215 }
216
217 #[test]
218 fn record_period() {
219 let mut latency = Latency::new();
220 assert_eq!(latency.periods(), 0);
221 assert!(latency.received().is_none());
222 assert!(latency.sent().is_none());
223 assert!(latency.recent().is_empty());
224
225 latency.record_sent();
226 assert_eq!(latency.periods(), 0);
227 assert!(latency.received().is_none());
228 assert!(latency.sent().is_some());
229
230 latency.record_received();
231 assert_eq!(latency.periods(), 1);
232 assert!(latency.received().is_some());
233 assert!(latency.sent().is_some());
234 assert_eq!(latency.recent().len(), 1);
235
236 latency.record_sent();
237 assert_eq!(latency.periods(), 1);
238 assert!(latency.received().is_none());
239 assert!(latency.sent().is_some());
240 assert_eq!(latency.recent().len(), 1);
241 }
242
243 #[test]
244 #[should_panic(expected = "period completed multiple times")]
245 fn record_completed_period() {
246 let mut latency = Latency::new();
247 latency.record_sent();
248 latency.record_received();
249 latency.record_received();
250 }
251
252 #[test]
253 #[should_panic(expected = "period has not begun")]
254 fn record_not_begun_period() {
255 let mut latency = Latency::new();
256 latency.record_received();
257 }
258}