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
extern crate pest;
#[macro_use]
extern crate pest_derive;

use pest::Parser;

mod metrics;

#[derive(Parser)]
#[grammar = "metric_grammar.pest"]
pub struct MetricParser;

pub fn parse_metrics(unparsed_metrics: String) {
    let raw_metrics = MetricParser::parse(Rule::metrics, &unparsed_metrics)
        .expect("unsuccessful parse 😥")
        .next().unwrap();

    //let mut metrics: HashMap<String, metrics::MetricGroup> = HashMap::new();
    let mut metrics = metrics::Metrics::new();

    for raw_metric in raw_metrics.into_inner() {
        let mut base_metric_name = String::new();
        for line in raw_metric.into_inner() {
            match line.as_rule() {
                Rule::helpLine => {
                    let mut inner = line.into_inner();
                    let metric_name = inner.next().unwrap().as_str().to_string();
                    base_metric_name = metric_name.clone();
                    let help_text = match inner.next() {
                        Some(t) => t.as_str(),
                        _ => "",
                    };
                    metrics.entry(metric_name)
                        .and_modify(|m| m.help = help_text.to_string())
                        .or_insert(metrics::MetricGroup::new_with_help(help_text));
                }
                Rule::typeLine => {
                    let mut inner = line.into_inner();
                    let metric_name = inner.next().unwrap().as_str().to_string();
                    base_metric_name = metric_name.clone();
                    let type_text = match inner.next() {
                        Some(t) => t.as_str(),
                        _ => "",
                    };
                    metrics.entry(metric_name)
                        .and_modify(
                            |m| m.r#type = match type_text {
                                "counter" => metrics::MetricType::COUNTER,
                                "gauge" => metrics::MetricType::GAUGE,
                                "histogram" => metrics::MetricType::HISTOGRAM,
                                "summary" => metrics::MetricType::SUMMARY,
                                _ => metrics::MetricType::NONE,
                            })
                        .or_insert(
                            metrics::MetricGroup::new_with_type(type_text));
                }
                Rule::metricLine => {
                    let mut inner = line.into_inner();
                    let metric_name = inner.next().unwrap().as_str();
                    let metric_text = match inner.next() {
                        Some(t) => t.as_str(),
                        _ => "",
                    };
                    metrics.entry(base_metric_name.clone())
                        .and_modify(
                            |m| m.metric.0.push(
                                metrics::add(metric_name, metric_text)))
                        .or_insert(
                            metrics::MetricGroup::new_with_metric(
                                metric_name, metric_text));
                }
                _ => {
        println!("{:#?}", line.into_inner());
                }

            }
        }
    }
    println!("{:#?}", metrics);
}