-
Notifications
You must be signed in to change notification settings - Fork 2
/
walking_stairs.rs
137 lines (118 loc) · 4.01 KB
/
walking_stairs.rs
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
126
127
128
129
130
131
132
133
134
135
136
137
use std::path::PathBuf;
use eyre::Result;
use plotters::prelude::*;
use extended_isolation_forest::{Forest, ForestOptions};
fn read_acceleration_data(
filename: &str,
apply_sliding_window: Option<usize>,
) -> eyre::Result<Vec<[f64; 3]>> {
let file_path: PathBuf = [env!("CARGO_MANIFEST_DIR"), "data", "acceleration", filename]
.iter()
.collect();
let mut csv_reader = csv::Reader::from_path(file_path.as_path())?;
let mut rows = Vec::new();
for record_res in csv_reader.records() {
let record = record_res?;
rows.push([
record[1].parse()?, // accel x
record[2].parse()?, // accel y
record[3].parse()?, // accel z
]);
}
if let Some(window_width) = apply_sliding_window {
let mut smoothend = Vec::new();
for window in rows.windows(window_width) {
smoothend.push([
window.iter().map(|v| v[0]).sum::<f64>() / window.len() as f64,
window.iter().map(|v| v[1]).sum::<f64>() / window.len() as f64,
window.iter().map(|v| v[2]).sum::<f64>() / window.len() as f64,
])
}
Ok(smoothend)
} else {
Ok(rows)
}
}
const OUT_FILE_NAME: &str = "walking_stairs.png";
fn main() -> Result<()> {
let smoothing = Some(20);
// train a forest
let forest = Forest::from_slice(
read_acceleration_data("walking-stairs.train.csv", smoothing)?.as_slice(),
&ForestOptions {
n_trees: 100,
sample_size: 600,
max_tree_depth: None,
extension_level: 1,
},
)?;
let rows = read_acceleration_data("walking-stairs.annomaly.csv", smoothing)?;
// plot results
let root = BitMapBackend::new(OUT_FILE_NAME, (1200, 800)).into_drawing_area();
root.fill(&WHITE)?;
let root = root.margin(10, 10, 10, 10);
let (upper, lower) = root.split_vertically(500);
let max_value = 3.0f64;
// plot the accelerations
let mut upper_chart = ChartBuilder::on(&upper)
.set_label_area_size(LabelAreaPosition::Left, 60)
.set_label_area_size(LabelAreaPosition::Bottom, 30)
.caption("Acceleration while walking stairs", ("sans-serif", 14))
.build_cartesian_2d(
0.0..(rows.len() as f64 - 1.0),
(max_value * -1.0)..max_value,
)?;
upper_chart
.configure_mesh()
.disable_mesh()
.y_desc("Linear Acceleration (m/s²)")
.x_desc("Time step")
.draw()?;
let line_styles = [
(0, &BLUE, "Accel(z)"),
(1, &GREEN, "Accel(y)"),
(2, &RED, "Accel(z)"),
//(3, &BLACK, "Accel(absolute)"),
];
for (array_idx, style, label) in line_styles {
upper_chart
.draw_series(LineSeries::new(
rows.iter()
.enumerate()
.map(|(idx, values)| (idx as f64, values[array_idx])),
style,
))?
.label(label)
.legend(move |(x, y)| PathElement::new(vec![(x, y), (x + 20, y)], style));
}
upper_chart
.configure_series_labels()
.background_style(RGBColor(200, 200, 200))
.draw()?;
let mut lower_chart = ChartBuilder::on(&lower)
.margin_top(10)
.set_label_area_size(LabelAreaPosition::Left, 60)
.set_label_area_size(LabelAreaPosition::Bottom, 30)
//.caption("Anomaly score", ("sans-serif", 14))
.build_cartesian_2d(0.0..(rows.len() as f64 - 1.0), 0.3..1.0)?;
lower_chart
.configure_mesh()
.disable_mesh()
.y_desc("Anomaly score")
.x_desc("Time step")
.draw()?;
// plot the detected anomalies
lower_chart.draw_series(
AreaSeries::new(
rows.iter()
.enumerate()
.map(|(idx, row)| (idx as f64, forest.score(row))),
0.0,
RED.mix(0.2),
)
.border_style(RED),
)?;
root.present()?;
println!("Result has been saved to {}", OUT_FILE_NAME);
Ok(())
}