glean_core/metrics/
string.rs1use std::sync::Arc;
6
7use crate::common_metric_data::{CommonMetricDataInternal, MetricLabel};
8use crate::error_recording::{test_get_num_recorded_errors, ErrorType};
9use crate::metrics::Metric;
10use crate::metrics::MetricType;
11use crate::util::truncate_string_at_boundary_with_error;
12use crate::Glean;
13use crate::{CommonMetricData, TestGetValue};
14
15pub const MAX_LENGTH_VALUE: usize = 255;
16
17#[derive(Clone, Debug)]
22pub struct StringMetric {
23 meta: Arc<CommonMetricDataInternal>,
24}
25
26impl MetricType for StringMetric {
27 fn meta(&self) -> &CommonMetricDataInternal {
28 &self.meta
29 }
30
31 fn with_name(&self, name: String) -> Self {
32 let mut meta = (*self.meta).clone();
33 meta.inner.name = name;
34 Self {
35 meta: Arc::new(meta),
36 }
37 }
38
39 fn with_label(&self, label: MetricLabel) -> Self {
40 let mut meta = (*self.meta).clone();
41 meta.inner.label = Some(label);
42 Self {
43 meta: Arc::new(meta),
44 }
45 }
46}
47
48impl StringMetric {
53 pub fn new(meta: CommonMetricData) -> Self {
55 Self {
56 meta: Arc::new(meta.into()),
57 }
58 }
59
60 pub fn set(&self, value: String) {
70 let metric = self.clone();
71 crate::launch_with_glean(move |glean| metric.set_sync(glean, &value))
72 }
73
74 #[doc(hidden)]
76 pub fn set_sync<S: Into<String>>(&self, glean: &Glean, value: S) {
77 if !self.should_record(glean) {
78 return;
79 }
80
81 let s = truncate_string_at_boundary_with_error(glean, &self.meta, value, MAX_LENGTH_VALUE);
82
83 let value = Metric::String(s);
84 glean.storage().record(glean, &self.meta, &value)
85 }
86
87 #[doc(hidden)]
89 pub fn get_value<'a, S: Into<Option<&'a str>>>(
90 &self,
91 glean: &Glean,
92 ping_name: S,
93 ) -> Option<String> {
94 let queried_ping_name = ping_name
95 .into()
96 .unwrap_or_else(|| &self.meta().inner.send_in_pings[0]);
97
98 match glean.storage().get_metric(self.meta(), queried_ping_name) {
99 Some(Metric::String(s)) => Some(s),
100 _ => None,
101 }
102 }
103
104 pub fn test_get_num_recorded_errors(&self, error: ErrorType) -> i32 {
116 crate::block_on_dispatcher();
117
118 crate::core::with_glean(|glean| {
119 test_get_num_recorded_errors(glean, self.meta(), error).unwrap_or(0)
120 })
121 }
122}
123
124impl TestGetValue for StringMetric {
125 type Output = String;
126
127 fn test_get_value(&self, ping_name: Option<String>) -> Option<String> {
142 crate::block_on_dispatcher();
143 crate::core::with_glean(|glean| self.get_value(glean, ping_name.as_deref()))
144 }
145}
146
147#[cfg(test)]
148mod test {
149 use super::*;
150 use crate::tests::new_glean;
151 use crate::util::truncate_string_at_boundary;
152 use crate::Lifetime;
153
154 #[test]
155 fn setting_a_long_string_records_an_error() {
156 let (glean, _t) = new_glean(None);
157
158 let metric = StringMetric::new(CommonMetricData {
159 name: "string_metric".into(),
160 category: "test".into(),
161 send_in_pings: vec!["store1".into()],
162 lifetime: Lifetime::Application,
163 disabled: false,
164 label: None,
165 in_session: false,
166 });
167
168 let sample_string = "0123456789".repeat(26);
169 metric.set_sync(&glean, sample_string.clone());
170
171 let truncated = truncate_string_at_boundary(sample_string, MAX_LENGTH_VALUE);
172 assert_eq!(truncated, metric.get_value(&glean, "store1").unwrap());
173
174 assert_eq!(
175 1,
176 test_get_num_recorded_errors(&glean, metric.meta(), ErrorType::InvalidOverflow)
177 .unwrap()
178 );
179 }
180}