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
use serde::Deserialize;
use tokio::time::Interval;
use super::Collect;
use crate::common::{ConfigInto, FromConfig, FromPath, Period, Render};
use async_trait::async_trait;
#[derive(Deserialize)]
pub struct TextCollectorConfig {
flush_period: Period,
separator: String,
}
impl FromPath for TextCollectorConfig {}
impl ConfigInto<TextCollector> for TextCollectorConfig {}
pub struct TextCollector {
buffer: String,
flush_period: Period,
separator: String,
}
#[async_trait]
impl FromConfig<TextCollectorConfig> for TextCollector {
async fn from_config(config: TextCollectorConfig) -> anyhow::Result<Self> {
Ok(TextCollector {
buffer: String::new(),
flush_period: config.flush_period,
separator: config.separator,
})
}
}
#[async_trait]
impl<T> Collect<T, String, TextCollectorConfig> for TextCollector
where
T: Render + Send + 'static,
{
async fn collect(&mut self, t: T) -> anyhow::Result<()> {
let text = t.render();
self.buffer.push_str(&text);
self.buffer.push_str(&self.separator);
Ok(())
}
async fn flush(&mut self) -> anyhow::Result<Option<String>> {
let buffer = self.buffer.clone();
self.buffer.clear();
if buffer.is_empty() {
return Ok(None);
}
Ok(Some(buffer))
}
fn get_flush_interval(&self) -> Interval {
let flush_period = self.flush_period.clone();
tokio::time::interval(flush_period.into())
}
}