use std::str::FromStr; use dtw_rs_band_fork::{Algorithm, DynamicTimeWarping, ParameterizedAlgorithm, Restriction}; use float_cmp::assert_approx_eq; #[test] fn dynamic_time_warping() { include_str!("dtw_test_cases.txt") .lines() .collect::>() .chunks_exact(5) .for_each(|chunk| { let a = into_float_vec::(chunk[0]); let b = into_float_vec::(chunk[1]); let expected_path = into_float_vec::(chunk[2]) .into_iter() .zip(into_float_vec::(chunk[3])) .collect::>(); let expected_distance = chunk[4].parse::().unwrap(); let dtw = DynamicTimeWarping::between(&a, &b); assert_approx_eq!(f64, dtw.distance(), expected_distance); assert_eq!(*dtw.path(), expected_path); }); } #[test] fn dynamic_time_warping_with_distance_closure() { let a = [1.0, 3.0, 9.0, 2.0, 1.0]; let b = [2.0, 0.0, 0.0, 8.0, 7.0, 2.0]; let expected_path = [(0, 0), (0, 1), (1, 2), (2, 3), (2, 4), (3, 5), (4, 5)]; let expected_distance = 9.0; let dtw = DynamicTimeWarping::with_closure(&a, &b, |a, b| f64::abs(a - b)); assert_eq!(dtw.distance(), expected_distance); assert_eq!(*dtw.path(), expected_path); } #[test] fn dynamic_time_warping_with_band_restriction() { let a = [1.0, 3.0, 9.0, 2.0, 1.0]; let b = [2.0, 0.0, 0.0, 8.0, 7.0, 2.0]; let expected_path = [(0, 0), (0, 1), (1, 2), (2, 3), (3, 4), (4, 5)]; let expected_distance = 12.0; let dtw = DynamicTimeWarping::with_param(&a, &b, Restriction::Band(1)); assert_eq!(dtw.distance(), expected_distance); assert_eq!(*dtw.path(), expected_path); } #[test] fn dynamic_time_warping_with_corner_out_of_band() { let a = [1.0, 3.0, 9.0, 2.0]; let b = [2.0, 0.0]; // The path should actually be cut off at (2, 1) let expected_path = [(0, 0), (1, 0), (2, 1), (3, 1)]; let expected_distance = 11.0; let dtw = DynamicTimeWarping::with_param(&a, &b, Restriction::Band(1)); assert_eq!(dtw.distance(), expected_distance); assert_eq!(*dtw.path(), expected_path); } #[test] fn dynamic_time_warping_with_band_restricted_and_distance_closure() { let a = [1.0, 3.0, 9.0, 2.0, 1.0]; let b = [2.0, 0.0, 0.0, 8.0, 7.0, 2.0]; let expected_path = [(0, 0), (0, 1), (1, 2), (2, 3), (3, 4), (4, 5)]; let expected_distance = 12.0; let dtw = DynamicTimeWarping::with_closure_and_param( &a, &b, |a, b| f64::abs(a - b), Restriction::Band(1), ); println!("{}", dtw); assert_eq!(dtw.distance(), expected_distance); assert_eq!(*dtw.path(), expected_path); } #[inline] fn into_float_vec(line: &str) -> Vec { line.split(' ') .map(str::parse::) .filter_map(Result::ok) .collect::>() }