Skip to main content

ballistics_engine/
wind.rs

1use nalgebra::Vector3;
2use std::f64::consts::PI;
3
4/// Conversion constant from KMH to MPS
5const KMH_TO_MPS: f64 = 1000.0 / 3600.0;
6
7/// Wind segment: (speed_kmh, angle_deg, until_distance_m)
8/// This matches the Python WindSock interface
9pub type WindSegment = (f64, f64, f64);
10
11/// Wind condition handler for trajectory calculations
12#[derive(Debug, Clone)]
13pub struct WindSock {
14    /// Sorted wind segments by distance
15    winds: Vec<WindSegment>,
16    /// Precomputed wind vector for each segment (parallel to `winds`). The Monte-Carlo RK4
17    /// kernel queries wind 4x per step, so caching avoids recomputing sin/cos every call.
18    wind_vecs: Vec<Vector3<f64>>,
19    /// Current segment index
20    current: usize,
21    /// Distance where next segment starts
22    next_range: f64,
23    /// Current wind vector
24    current_vec: Vector3<f64>,
25}
26
27impl WindSock {
28    /// Create a new WindSock from wind segments
29    ///
30    /// Args:
31    ///     segments: List of (speed_kmh, angle_deg, until_distance_m) tuples
32    pub fn new(mut segments: Vec<WindSegment>) -> Self {
33        // Sort segments by distance, handling NaN safely by treating it as greater than any value
34        segments.sort_by(|a, b| a.2.partial_cmp(&b.2).unwrap_or(std::cmp::Ordering::Greater));
35
36        // Precompute each segment's wind vector once (depends only on its speed/angle).
37        let wind_vecs: Vec<Vector3<f64>> = segments.iter().map(Self::calc_vec).collect();
38
39        let (current, next_range, current_vec) = if segments.is_empty() {
40            (0, f64::INFINITY, Vector3::zeros())
41        } else {
42            (0, segments[0].2, wind_vecs[0])
43        };
44
45        WindSock {
46            winds: segments,
47            wind_vecs,
48            current,
49            next_range,
50            current_vec,
51        }
52    }
53
54    /// Calculate wind vector from wind segment
55    fn calc_vec(seg: &WindSegment) -> Vector3<f64> {
56        let (speed_kmh, angle_deg, _) = *seg;
57
58        // Convert kmh to m/s
59        let speed_mps = speed_kmh * KMH_TO_MPS;
60        let angle_rad = angle_deg * PI / 180.0;
61
62        // Wind convention (matching trajectory coordinates):
63        // 0° = headwind (from front, affects -x downrange)
64        // 90° = wind from right (affects -z lateral)
65        // 180° = tailwind (from back, affects +x downrange)
66        // 270° = wind from left (affects +z lateral)
67        //
68        // McCoy convention: x=downrange, y=vertical, z=lateral
69        Vector3::new(
70            -speed_mps * angle_rad.cos(), // x (downrange - head/tail component)
71            0.0,                          // y (vertical)
72            -speed_mps * angle_rad.sin(), // z (lateral - crosswind component)
73        )
74    }
75
76    /// Get wind vector for a given range
77    ///
78    /// Note: This modifies internal state and expects monotonically increasing ranges
79    /// For trajectory integration, we need a stateless version
80    pub fn vector_for_range(&mut self, range_m: f64) -> Vector3<f64> {
81        // Handle NaN
82        if range_m.is_nan() {
83            return Vector3::zeros();
84        }
85
86        // Advance the cursor across however many segments the query skipped (a single `if`
87        // returned a stale vector when a monotonic query jumped past a whole short segment).
88        while range_m >= self.next_range && self.current < self.winds.len() {
89            self.current += 1;
90            if self.current >= self.winds.len() {
91                self.current_vec = Vector3::zeros();
92                self.next_range = f64::INFINITY;
93            } else {
94                self.current_vec = self.wind_vecs[self.current];
95                self.next_range = self.winds[self.current].2;
96            }
97        }
98
99        self.current_vec
100    }
101
102    /// Get wind vector for a given range (stateless version)
103    ///
104    /// This version doesn't modify internal state and is safe for numerical integration
105    /// where the same range might be queried multiple times or out of order
106    pub fn vector_for_range_stateless(&self, range_m: f64) -> Vector3<f64> {
107        // Handle NaN
108        if range_m.is_nan() {
109            return Vector3::zeros();
110        }
111
112        // Find the appropriate segment (precomputed vector — no per-call trig).
113        for (i, segment) in self.winds.iter().enumerate() {
114            if range_m < segment.2 {
115                return self.wind_vecs[i];
116            }
117        }
118
119        // Beyond all segments
120        Vector3::zeros()
121    }
122}
123
124/// Parse a `"SPEED:ANGLE:UNTIL_DISTANCE"` string into a [`WindSegment`]
125/// `(speed_kmh, angle_deg, until_distance_m)`.
126///
127/// `imperial`: when true, SPEED is mph and UNTIL_DISTANCE is yards; otherwise
128/// SPEED is m/s and UNTIL_DISTANCE is meters. ANGLE is always degrees in the
129/// wind-FROM convention (0 = headwind, 90 = from the right). Shared by the CLI
130/// (`--wind-segment`) and the WASM front-ends so they parse identically.
131pub fn parse_wind_segment_str(s: &str, imperial: bool) -> Result<WindSegment, String> {
132    let parts: Vec<&str> = s.split(':').collect();
133    if parts.len() != 3 {
134        return Err(format!(
135            "invalid wind segment '{s}': expected SPEED:ANGLE:UNTIL_DISTANCE (three colon-separated numbers)"
136        ));
137    }
138    let num = |i: usize, name: &str| -> Result<f64, String> {
139        parts[i].trim().parse::<f64>().map_err(|_| {
140            format!("invalid wind segment '{s}': {name} '{}' is not a number", parts[i])
141        })
142    };
143    let speed = num(0, "speed")?;
144    let angle = num(1, "angle")?;
145    let until = num(2, "until-distance")?;
146    if !speed.is_finite() || !angle.is_finite() || !until.is_finite() {
147        return Err(format!(
148            "invalid wind segment '{s}': speed, angle, and until-distance must be finite numbers"
149        ));
150    }
151    if speed < 0.0 {
152        return Err(format!("invalid wind segment '{s}': speed must be >= 0"));
153    }
154    if until <= 0.0 {
155        return Err(format!("invalid wind segment '{s}': until-distance must be > 0"));
156    }
157    let (speed_kmh, until_m) = if imperial {
158        (speed * 1.609344, until * 0.9144) // mph -> km/h, yards -> meters
159    } else {
160        (speed * 3.6, until) // m/s -> km/h, meters -> meters
161    };
162    Ok((speed_kmh, angle, until_m))
163}
164
165#[cfg(test)]
166mod tests {
167    use super::*;
168
169    #[test]
170    fn test_wind_sock_empty() {
171        let sock = WindSock::new(vec![]);
172        assert_eq!(sock.vector_for_range_stateless(50.0), Vector3::zeros());
173    }
174
175    #[test]
176    fn test_wind_sock_single_segment() {
177        // 16.0934 kmh (10 mph) @ 90° until 100m
178        let sock = WindSock::new(vec![(16.0934, 90.0, 100.0)]);
179
180        // Should have wind before 100m
181        let vec_50 = sock.vector_for_range_stateless(50.0);
182        println!("vec_50 = [{}, {}, {}]", vec_50[0], vec_50[1], vec_50[2]);
183        assert!(vec_50.norm() > 0.0);
184        // 90° wind from right (crosswind, McCoy): negative Z (lateral), zero Y, near-zero X (downrange)
185        assert!(
186            vec_50[2] < 0.0,
187            "Z (lateral) should be negative for 90° wind, got {}",
188            vec_50[2]
189        );
190        assert_eq!(vec_50[1], 0.0); // Zero Y component
191        assert!(
192            vec_50[0].abs() < 0.01,
193            "X (downrange) should be nearly zero for 90° wind, got {}",
194            vec_50[0]
195        );
196
197        // No wind after 100m
198        let vec_150 = sock.vector_for_range_stateless(150.0);
199        assert_eq!(vec_150, Vector3::zeros());
200    }
201
202    #[test]
203    fn test_wind_sock_multiple_segments() {
204        // Multiple wind segments (in kmh)
205        let sock = WindSock::new(vec![
206            (16.0934, 90.0, 50.0),  // 10 mph @ 90° until 50m
207            (24.1401, 45.0, 100.0), // 15 mph @ 45° until 100m
208            (8.0467, 180.0, 200.0), // 5 mph @ 180° until 200m
209        ]);
210
211        // Test each segment
212        let vec_25 = sock.vector_for_range_stateless(25.0);
213        println!("vec_25 = [{}, {}, {}]", vec_25[0], vec_25[1], vec_25[2]);
214        assert!(vec_25.norm() > 0.0);
215        assert!(vec_25[2] < 0.0, "90° wind should have negative Z (lateral)"); // 90° wind from right
216
217        let vec_75 = sock.vector_for_range_stateless(75.0);
218        println!("vec_75 = [{}, {}, {}]", vec_75[0], vec_75[1], vec_75[2]);
219        assert!(vec_75.norm() > vec_25.norm()); // 15 mph > 10 mph
220        assert!(vec_75[0] < 0.0); // 45° wind has negative X component
221        assert!(vec_75[2] < 0.0); // 45° wind has negative Z component
222
223        let vec_150 = sock.vector_for_range_stateless(150.0);
224        println!("vec_150 = [{}, {}, {}]", vec_150[0], vec_150[1], vec_150[2]);
225        assert!(vec_150.norm() < vec_75.norm()); // 5 mph < 15 mph
226        assert!(
227            vec_150[2].abs() < 0.01,
228            "180° wind should have near-zero Z (lateral), got {}",
229            vec_150[2]
230        ); // 180° wind (from behind)
231        assert!(
232            vec_150[0] > 0.0,
233            "180° wind should have positive X (tailwind, downrange), got {}",
234            vec_150[0]
235        ); // Tailwind
236
237        let vec_250 = sock.vector_for_range_stateless(250.0);
238        assert_eq!(vec_250, Vector3::zeros()); // Beyond all segments
239    }
240
241    #[test]
242    fn test_wind_conversion() {
243        // Test conversion: 16.0934 km/h = 4.47 m/s
244        let sock = WindSock::new(vec![(16.0934, 0.0, 100.0)]);
245        let vec = sock.vector_for_range_stateless(50.0);
246
247        let expected_speed = 16.0934 * KMH_TO_MPS;
248        assert!((vec.norm() - expected_speed).abs() < 0.01);
249    }
250
251    #[test]
252    fn test_wind_sock_boundary_is_upper_exclusive() {
253        // A segment's `until_distance_m` is exclusive: a query exactly at the
254        // boundary rolls to the next segment.
255        let sock = WindSock::new(vec![(16.0934, 90.0, 100.0), (32.1868, 270.0, 200.0)]);
256        // Just below 100 m -> first segment (90deg, negative Z).
257        assert!(sock.vector_for_range_stateless(99.999)[2] < 0.0);
258        // Exactly 100 m -> second segment (270deg, positive Z).
259        assert!(sock.vector_for_range_stateless(100.0)[2] > 0.0);
260        // Beyond the last boundary -> zero.
261        assert_eq!(sock.vector_for_range_stateless(200.0), Vector3::zeros());
262    }
263
264    #[test]
265    fn test_parse_wind_segment_str_units() {
266        // Imperial: 10 mph -> 16.0934 km/h, 100 yd -> 91.44 m.
267        let (kmh, ang, until) = parse_wind_segment_str("10:90:100", true).unwrap();
268        assert!((kmh - 16.09344).abs() < 1e-4);
269        assert_eq!(ang, 90.0);
270        assert!((until - 91.44).abs() < 1e-4);
271
272        // Metric: 5 m/s -> 18 km/h, 200 m stays 200 m.
273        let (kmh, ang, until) = parse_wind_segment_str("5:270:200", false).unwrap();
274        assert!((kmh - 18.0).abs() < 1e-9);
275        assert_eq!(ang, 270.0);
276        assert!((until - 200.0).abs() < 1e-9);
277
278        // Malformed inputs are rejected.
279        assert!(parse_wind_segment_str("10:90", true).is_err()); // too few fields
280        assert!(parse_wind_segment_str("10:bad:100", true).is_err()); // non-numeric
281        assert!(parse_wind_segment_str("10:90:0", true).is_err()); // zero until-distance
282        assert!(parse_wind_segment_str("-3:90:100", true).is_err()); // negative speed
283        // Non-finite values must be rejected (NaN comparisons would slip past < / <=).
284        assert!(parse_wind_segment_str("10:nan:5000", true).is_err());
285        assert!(parse_wind_segment_str("10:90:nan", true).is_err());
286        assert!(parse_wind_segment_str("inf:90:100", true).is_err());
287    }
288}